Files
evgeniyChepelev 315283822c iOS home screen widget (#78)
* Add Home Screen widget with VPN toggle and refactor app activation logic

- Add NetBirdWidgetExtension target with small/medium widget sizes
- Support direct connect/disconnect from widget via interactive buttons (iOS 17+)
- Detect missing VPN config or login-required state and open app via deep link
- Poll for stable VPN state after toggle to prevent loader getting stuck
- Add widget shared state keys to GlobalConstants and sync status from MainViewModel
- Fix false "authentication required" alert on app resume after widget disconnect
- Deduplicate app activation logic into shared startActivation/stopActivation
- Extract polling helpers: updateDetailsIfChanged, updatePeersIfChanged, applyExtensionStatus
2026-04-14 10:30:08 +02:00

808 lines
29 KiB
Swift

//
// MainViewModel.swift
// NetBirdiOS
//
// Created by Pascal Fischer on 01.08.23.
//
// This ViewModel is shared between iOS and tvOS.
// Platform-specific code is wrapped with #if os() directives.
//
import SwiftUI
import NetworkExtension
import Network
import os
import Combine
import NetBirdSDK
import UserNotifications
#if os(iOS)
import WidgetKit
#endif
/// Used by updateManagementURL to check if SSO is supported
class SSOCheckListener: NSObject, NetBirdSDKSSOListenerProtocol {
var onResult: ((Bool?, Error?) -> Void)?
func onError(_ p0: Error?) {
onResult?(nil, p0)
}
func onSuccess(_ p0: Bool) {
onResult?(p0, nil)
}
}
// Error Listener for setup key login
/// Used by setSetupKey to handle async login result
class SetupKeyErrListener: NSObject, NetBirdSDKErrListenerProtocol {
var onResult: ((Error?) -> Void)?
func onError(_ p0: Error?) {
onResult?(p0)
}
func onSuccess() {
onResult?(nil)
}
}
enum VPNDisplayState {
case connected
case connecting
case disconnecting
case disconnected
}
/// For both iOS and tvOS (tvOS 17+ required for VPN support).
@MainActor
class ViewModel: ObservableObject {
private let logger = Logger(subsystem: "io.netbird.app", category: "ViewModel")
// VPN Adapter (shared)
@Published var networkExtensionAdapter: NetworkExtensionAdapter
// UI State (shared)
@Published var showSetupKeyPopup = false
@Published var showChangeServerAlert = false
@Published var showInvalidServerAlert = false
@Published var showInvalidSetupKeyHint = false
@Published var showInvalidSetupKeyAlert = false
@Published var showLogLevelChangedAlert = false
@Published var showBetaProgramAlert = false
@Published var showInvalidPresharedKeyAlert = false
@Published var showServerChangedInfo = false
@Published var showPreSharedKeyChangedInfo = false
@Published var showFqdnCopiedAlert = false
@Published var showIpCopiedAlert = false
@Published var showAuthenticationRequired = false
@Published var navigateToServerView = false
@Published var extensionState: NEVPNStatus = .disconnected
@Published var managementStatus: ClientState = .disconnected
@Published var statusDetailsValid = false
@Published var extensionStateText = "Disconnected"
@Published var vpnDisplayState: VPNDisplayState = .disconnected
var connectPressed = false
var disconnectPressed = false
@Published var rosenpassEnabled = false
@Published var rosenpassPermissive = false
@Published var presharedKey = ""
@Published var server: String = ""
@Published var setupKey: String = ""
@Published var presharedKeySecure = true
@Published var fqdn = UserDefaults.standard.string(forKey: "fqdn") ?? ""
@Published var ip = UserDefaults.standard.string(forKey: "ip") ?? ""
// Debug
@Published var traceLogsEnabled: Bool {
didSet {
self.showLogLevelChangedAlert = true
DispatchQueue.main.asyncAfter(deadline: .now() + 3) {
self.showLogLevelChangedAlert = false
}
let logLevel = traceLogsEnabled ? "TRACE" : "INFO"
UserDefaults.standard.set(logLevel, forKey: "logLevel")
UserDefaults.standard.synchronize()
}
}
@Published var forceRelayConnection = true
@Published var showForceRelayAlert = false
@Published var connectOnDemand = false
@Published var showOnDemandAlert = false
@Published var showOnDemandConflictAlert = false
@Published var showOnDemandDisconnectAlert = false
@Published var onDemandWiFiPolicy: WiFiOnDemandPolicy = .always
@Published var onDemandCellularPolicy: CellularOnDemandPolicy = .always
@Published var onDemandWiFiNetworks: [String] = []
@Published var knownSSIDs: [String] = []
@Published var showRosenpassChangedAlert = false
@Published var networkUnavailable = false
@Published var isInternetConnected = true
/// Platform-agnostic configuration provider.
/// Abstracts iOS SDK preferences vs tvOS UserDefaults + IPC.
private lazy var configProvider: ConfigurationProvider = ConfigurationProviderFactory.create()
var buttonLock = false
let defaults = UserDefaults.standard
private var cancellables = Set<AnyCancellable>()
private let networkMonitor = NWPathMonitor()
private let monitorQueue = DispatchQueue(label: "io.netbird.networkMonitor")
@Published var peerViewModel: PeerViewModel
@Published var routeViewModel: RoutesViewModel
init() {
let networkExtensionAdapter = NetworkExtensionAdapter()
self.networkExtensionAdapter = networkExtensionAdapter
let logLevel = UserDefaults.standard.string(forKey: "logLevel") ?? "INFO"
self.traceLogsEnabled = logLevel == "TRACE"
self.peerViewModel = PeerViewModel()
self.routeViewModel = RoutesViewModel(networkExtensionAdapter: networkExtensionAdapter)
// Don't load rosenpass settings during init - they trigger expensive SDK initialization.
// These will be loaded lazily when the settings view is accessed.
// self.rosenpassEnabled = self.getRosenpassEnabled()
// self.rosenpassPermissive = self.getRosenpassPermissive()
// forceRelayConnection uses UserDefaults (not SDK), so it's safe to load during init
self.forceRelayConnection = self.getForcedRelayConnectionEnabled()
self.connectOnDemand = self.getConnectOnDemandEnabled()
self.loadOnDemandSettings()
networkMonitor.pathUpdateHandler = { [weak self] path in
DispatchQueue.main.async {
self?.isInternetConnected = path.status == .satisfied
self?.updateVPNDisplayState()
}
}
networkMonitor.start(queue: monitorQueue)
$setupKey
.removeDuplicates()
.debounce(for: .seconds(0.5), scheduler: RunLoop.main)
.map { setupKey in
!self.isValidSetupKey(setupKey)
}
.assign(to: &$showInvalidSetupKeyHint)
}
func connect() {
logger.info("connect: ENTRY POINT - function called")
#if os(iOS)
// Check if On Demand rules would block the connection on the current interface
if connectOnDemand && !onDemandRulesAllowConnect() {
logger.info("connect: On Demand rules conflict with current network, showing alert")
showOnDemandConflictAlert = true
return
}
#endif
performConnect()
}
/// Performs the actual VPN connection (called directly or after user dismisses On Demand conflict).
func performConnect() {
self.connectPressed = true
self.buttonLock = true
// Reset networkUnavailable flag when user initiates connection
self.networkUnavailable = false
#if os(iOS)
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
userDefaults?.set(false, forKey: GlobalConstants.keyNetworkUnavailable)
userDefaults?.synchronize()
#endif
updateVPNDisplayState()
logger.info("connect: connectPressed=true, buttonLock=true, starting adapter...")
// Reset buttonLock after 3 seconds
DispatchQueue.main.asyncAfter(deadline: .now() + 3) {
self.buttonLock = false
}
// Start the VPN connection
Task {
self.logger.info("connect: Task started, calling networkExtensionAdapter.start()")
await self.networkExtensionAdapter.start()
self.logger.info("connect: networkExtensionAdapter.start() completed")
}
}
/// Disables On Demand and connects (user chose to override conflicting rules).
func connectWithOnDemandDisabled() {
setConnectOnDemand(isEnabled: false)
performConnect()
}
#if os(iOS)
/// Checks whether On Demand rules would allow a connection on the current network interface.
/// Uses NWPathMonitor snapshot and current SSID to evaluate against saved policies.
private func onDemandRulesAllowConnect() -> Bool {
let path = networkMonitor.currentPath
// Determine which interface is active
let isOnWiFi = path.usesInterfaceType(.wifi)
let isOnCellular = path.usesInterfaceType(.cellular)
if isOnWiFi {
switch onDemandWiFiPolicy {
case .never:
return false
case .onlyOn:
guard let currentSSID = getCurrentSSID(), !currentSSID.isEmpty else {
return false
}
return onDemandWiFiNetworks.contains(currentSSID)
case .exceptOn:
guard let currentSSID = getCurrentSSID(), !currentSSID.isEmpty else {
return true
}
return !onDemandWiFiNetworks.contains(currentSSID)
case .always, .doNothing:
return true
}
}
if isOnCellular {
switch onDemandCellularPolicy {
case .never:
return false
case .always, .doNothing:
return true
}
}
return true
}
/// Checks whether On Demand has an active connect rule that will reconnect the tunnel
/// after a manual disconnect. Unlike onDemandRulesAllowConnect(), this excludes .doNothing
/// and only evaluates the currently active interface.
private func onDemandWillReconnect() -> Bool {
let path = networkMonitor.currentPath
let isOnWiFi = path.usesInterfaceType(.wifi)
let isOnCellular = path.usesInterfaceType(.cellular)
if isOnWiFi {
switch onDemandWiFiPolicy {
case .always:
return true
case .onlyOn:
guard let currentSSID = getCurrentSSID(), !currentSSID.isEmpty else {
return false
}
return onDemandWiFiNetworks.contains(currentSSID)
case .exceptOn:
guard let currentSSID = getCurrentSSID(), !currentSSID.isEmpty else {
return false
}
return !onDemandWiFiNetworks.contains(currentSSID)
case .never, .doNothing:
return false
}
} else if isOnCellular {
switch onDemandCellularPolicy {
case .always:
return true
case .never, .doNothing:
return false
}
}
return false
}
private func getCurrentSSID() -> String? {
// Synchronous check not possible with NEHotspotNetwork.fetchCurrent()
// Use cached value from last fetch if available
return _cachedSSID
}
private var _cachedSSID: String?
func refreshCurrentSSID() {
NEHotspotNetwork.fetchCurrent { [weak self] network in
DispatchQueue.main.async {
self?._cachedSSID = network?.ssid
}
}
}
#endif
func close() -> Void {
#if os(iOS)
// Warn user that On Demand will reconnect if rules match
if connectOnDemand && onDemandRulesAllowConnect() {
showOnDemandDisconnectAlert = true
return
}
#endif
performClose()
}
/// Performs the actual VPN disconnect.
func performClose() {
self.disconnectPressed = true
DispatchQueue.main.async {
print("Stopping extension")
self.buttonLock = true
DispatchQueue.main.asyncAfter(deadline: .now() + 3) {
self.buttonLock = false
}
self.networkExtensionAdapter.stop()
self.updateVPNDisplayState()
}
}
/// Disables On Demand and disconnects (user chose to prevent auto-reconnect).
func closeWithOnDemandDisabled() {
setConnectOnDemand(isEnabled: false)
performClose()
}
func updateVPNDisplayState() {
let newState: VPNDisplayState
// Extension state is the source of truth.
// Flags only provide immediate UI feedback for the brief gap
// between button press and extension state change.
switch extensionState {
case .connected:
// Extension confirmed connected clear both flags
connectPressed = false
disconnectPressed = false
newState = .connected
case .connecting:
connectPressed = false
newState = .connecting
case .disconnecting:
disconnectPressed = false
newState = .disconnecting
case .disconnected:
// Extension confirmed disconnected clear both flags,
// unless a flag was JUST set (immediate feedback)
if connectPressed {
newState = .connecting
} else {
disconnectPressed = false
newState = .disconnected
}
default:
connectPressed = false
disconnectPressed = false
newState = .disconnected
}
vpnDisplayState = newState
switch newState {
case .connected:
extensionStateText = isInternetConnected ? "Connected" : "Offline"
case .connecting:
extensionStateText = "Connecting..."
case .disconnecting:
extensionStateText = "Disconnecting..."
case .disconnected:
extensionStateText = "Disconnected"
}
#if os(iOS)
updateWidgetState()
#endif
}
#if os(iOS)
/// Writes current VPN state to shared UserDefaults so the widget can read it.
private func updateWidgetState() {
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
let statusString: String
switch vpnDisplayState {
case .connected: statusString = "connected"
case .connecting: statusString = "connecting"
case .disconnecting: statusString = "disconnecting"
case .disconnected: statusString = "disconnected"
}
userDefaults?.set(statusString, forKey: GlobalConstants.keyWidgetVPNStatus)
userDefaults?.set(ip, forKey: GlobalConstants.keyWidgetIP)
userDefaults?.set(fqdn, forKey: GlobalConstants.keyWidgetFQDN)
WidgetCenter.shared.reloadAllTimelines()
}
#endif
func startPollingDetails() {
#if os(iOS)
refreshCurrentSSID()
#endif
networkExtensionAdapter.startTimer { details in
self.checkExtensionState()
self.checkNetworkUnavailableFlag()
self.updateDetailsIfChanged(details)
self.updatePeersIfChanged(details)
self.statusDetailsValid = true
}
}
private func updateDetailsIfChanged(_ details: StatusDetails) {
let ipChanged = details.ip != ip
let fqdnChanged = details.fqdn != fqdn
let statusChanged = details.managementStatus != managementStatus
guard ipChanged || fqdnChanged || statusChanged else { return }
if ipChanged {
defaults.set(details.ip, forKey: "ip")
ip = details.ip
}
if fqdnChanged {
defaults.set(details.fqdn, forKey: "fqdn")
fqdn = details.fqdn
}
if statusChanged {
managementStatus = details.managementStatus
updateVPNDisplayState()
} else if ipChanged || fqdnChanged {
#if os(iOS)
updateWidgetState()
#endif
}
}
private func updatePeersIfChanged(_ details: StatusDetails) {
let sorted = details.peerInfo.sorted { $0.ip < $1.ip }
let current = peerViewModel.peerInfo
let changed = sorted.count != current.count || !sorted.elementsEqual(current) { a, b in
a.ip == b.ip
&& a.connStatus == b.connStatus
&& a.relayed == b.relayed
&& a.direct == b.direct
&& a.connStatusUpdate == b.connStatusUpdate
&& a.routes.count == b.routes.count
}
if changed {
peerViewModel.peerInfo = sorted
}
}
func stopPollingDetails() {
networkExtensionAdapter.stopTimer()
}
func checkExtensionState() {
networkExtensionAdapter.getExtensionStatus { status in
DispatchQueue.main.async {
self.applyExtensionStatus(status)
}
}
}
private func applyExtensionStatus(_ status: NEVPNStatus) {
let knownStatuses: Set<NEVPNStatus> = [.connected, .disconnected, .connecting, .disconnecting]
guard knownStatuses.contains(status), extensionState != status else { return }
extensionState = status
updateVPNDisplayState()
if status == .connected {
routeViewModel.getRoutes()
if connectOnDemand {
networkExtensionAdapter.setOnDemandEnabled(true)
}
}
}
func clearDetails() {
self.ip = ""
self.fqdn = ""
defaults.removeObject(forKey: "ip")
defaults.removeObject(forKey: "fqdn")
// Disable and persist On Demand off to keep UI/storage/manager in sync
setConnectOnDemand(isEnabled: false)
// Clear config JSON (contains server credentials and all settings)
Preferences.removeConfigFromUserDefaults()
// Reset @Published properties to reflect cleared state in UI
self.rosenpassEnabled = false
self.rosenpassPermissive = false
self.presharedKey = ""
self.presharedKeySecure = false
#if os(tvOS)
// Also clear extension-local config to prevent stale credentials
networkExtensionAdapter.clearExtensionConfig()
#endif
}
// MARK: - Configuration Methods (via ConfigurationProvider)
func updatePreSharedKey() {
configProvider.preSharedKey = presharedKey
if configProvider.commit() {
self.close()
self.presharedKeySecure = true
self.showPreSharedKeyChangedInfo = true
} else {
print("Failed to update preshared key")
}
}
func removePreSharedKey() {
presharedKey = ""
configProvider.preSharedKey = ""
if configProvider.commit() {
self.close()
self.presharedKeySecure = false
} else {
print("Failed to remove preshared key")
}
}
func loadPreSharedKey() {
self.presharedKey = configProvider.preSharedKey
self.presharedKeySecure = configProvider.hasPreSharedKey
}
func setRosenpassEnabled(enabled: Bool) {
// Update @Published property for immediate UI feedback
self.rosenpassEnabled = enabled
// Persist to storage (on tvOS this writes directly to config JSON)
configProvider.rosenpassEnabled = enabled
if !configProvider.commit() {
print("Failed to update rosenpass settings")
}
#if os(tvOS)
// Show reconnect alert if currently connected
if extensionState == .connected {
showRosenpassChangedAlert = true
}
#endif
}
func getRosenpassEnabled() -> Bool {
return configProvider.rosenpassEnabled
}
func getRosenpassPermissive() -> Bool {
return configProvider.rosenpassPermissive
}
/// Loads Rosenpass settings from the configuration provider into the @Published properties.
/// Call this when opening settings views to sync UI with stored values.
/// On iOS, this triggers SDK initialization, so it's deferred until needed.
/// On tvOS, this reads from UserDefaults which is fast.
func loadRosenpassSettings() {
self.rosenpassEnabled = configProvider.rosenpassEnabled
self.rosenpassPermissive = configProvider.rosenpassPermissive
}
func setRosenpassPermissive(permissive: Bool) {
// Update @Published property for immediate UI feedback
self.rosenpassPermissive = permissive
// Persist to storage (on tvOS this writes directly to config JSON)
configProvider.rosenpassPermissive = permissive
if !configProvider.commit() {
print("Failed to update rosenpass permissive settings")
}
}
/// Reloads configuration from persistent storage.
/// Call this after server changes or when returning to settings view.
func reloadConfiguration() {
configProvider.reload()
// Sync @Published properties with reloaded config values
loadRosenpassSettings()
}
func setForcedRelayConnection(isEnabled: Bool) {
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
userDefaults?.set(isEnabled, forKey: GlobalConstants.keyForceRelayConnection)
self.forceRelayConnection = isEnabled
self.showForceRelayAlert = true
}
func getForcedRelayConnectionEnabled() -> Bool {
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
#if os(iOS)
userDefaults?.register(defaults: [GlobalConstants.keyForceRelayConnection: true])
return userDefaults?.bool(forKey: GlobalConstants.keyForceRelayConnection) ?? true
#else
// forced relay battery optimization not need on Apple Tv
userDefaults?.register(defaults: [GlobalConstants.keyForceRelayConnection: false])
return userDefaults?.bool(forKey: GlobalConstants.keyForceRelayConnection) ?? false
#endif
}
func setConnectOnDemand(isEnabled: Bool) {
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
userDefaults?.set(isEnabled, forKey: GlobalConstants.keyConnectOnDemand)
self.connectOnDemand = isEnabled
networkExtensionAdapter.setOnDemandEnabled(isEnabled)
if isEnabled {
self.showOnDemandAlert = true
}
}
func getConnectOnDemandEnabled() -> Bool {
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
return userDefaults?.bool(forKey: GlobalConstants.keyConnectOnDemand) ?? false
}
func loadOnDemandSettings() {
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
let wifiRaw = userDefaults?.string(forKey: GlobalConstants.keyOnDemandWiFiPolicy) ?? WiFiOnDemandPolicy.always.rawValue
let cellularRaw = userDefaults?.string(forKey: GlobalConstants.keyOnDemandCellularPolicy) ?? CellularOnDemandPolicy.always.rawValue
self.onDemandWiFiPolicy = WiFiOnDemandPolicy(rawValue: wifiRaw) ?? .always
self.onDemandCellularPolicy = CellularOnDemandPolicy(rawValue: cellularRaw) ?? .always
self.onDemandWiFiNetworks = userDefaults?.stringArray(forKey: GlobalConstants.keyOnDemandWiFiNetworks) ?? []
self.knownSSIDs = userDefaults?.stringArray(forKey: GlobalConstants.keyKnownSSIDs) ?? []
}
func saveOnDemandSettings() {
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
userDefaults?.set(onDemandWiFiPolicy.rawValue, forKey: GlobalConstants.keyOnDemandWiFiPolicy)
userDefaults?.set(onDemandCellularPolicy.rawValue, forKey: GlobalConstants.keyOnDemandCellularPolicy)
userDefaults?.set(onDemandWiFiNetworks, forKey: GlobalConstants.keyOnDemandWiFiNetworks)
if connectOnDemand {
networkExtensionAdapter.applyOnDemandRules(
wifiPolicy: onDemandWiFiPolicy,
cellularPolicy: onDemandCellularPolicy,
wifiNetworks: onDemandWiFiNetworks
)
}
}
func addOnDemandWiFiNetwork(_ ssid: String) {
let trimmed = ssid.trimmingCharacters(in: .whitespacesAndNewlines)
guard !trimmed.isEmpty, !onDemandWiFiNetworks.contains(trimmed) else { return }
onDemandWiFiNetworks.append(trimmed)
saveOnDemandSettings()
}
func removeOnDemandWiFiNetwork(at offsets: IndexSet) {
onDemandWiFiNetworks.remove(atOffsets: offsets)
saveOnDemandSettings()
}
func recordKnownSSID(_ ssid: String) {
let trimmed = ssid.trimmingCharacters(in: .whitespacesAndNewlines)
guard !trimmed.isEmpty, !knownSSIDs.contains(trimmed) else { return }
knownSSIDs.append(trimmed)
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
userDefaults?.set(knownSSIDs, forKey: GlobalConstants.keyKnownSSIDs)
}
func removeKnownSSID(_ ssid: String) {
knownSSIDs.removeAll { $0 == ssid }
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
userDefaults?.set(knownSSIDs, forKey: GlobalConstants.keyKnownSSIDs)
}
func getDefaultStatus() -> StatusDetails {
return StatusDetails(ip: "", fqdn: "", managementStatus: .disconnected, peerInfo: [])
}
func isValidSetupKey(_ string: String) -> Bool {
if string.isEmpty { return true }
let pattern = "^[0-9A-Fa-f]{8}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{12}$"
let isMatch = string.range(of: pattern, options: .regularExpression, range: nil, locale: nil) != nil
return isMatch
}
func printLogContents(from logURL: URL) {
do {
let logContents = try String(contentsOf: logURL, encoding: .utf8)
print(logContents)
} catch {
print("Failed to read the log file: \(error.localizedDescription)")
}
}
/// Handles server change completion by stopping the engine and resetting all connection state.
func handleServerChanged() {
AppLogger.shared.log("Server changed - stopping engine and resetting state")
// Stop polling to prevent transitional states from updating UI
stopPollingDetails()
// Reset connection flags first to update UI immediately
connectPressed = false
disconnectPressed = false
buttonLock = false
// Reset connection state
extensionState = .disconnected
managementStatus = .disconnected
updateVPNDisplayState()
// Clear peer info
peerViewModel.peerInfo = []
// Clear connection details
clearDetails()
// Stop the network extension in background (non-blocking)
Task { @MainActor in
self.networkExtensionAdapter.stop()
}
// Reload configuration for new server
reloadConfiguration()
}
/// Checks shared app-group container for network unavailable flag set by the network extension.
/// Updates the networkUnavailable property to trigger UI animation changes.
/// iOS only - tvOS has a platform limitation where `UserDefaults(suiteName:)` does not
/// reliably synchronize between the main app and network extension processes, even with
/// a correctly configured App Group. On tvOS, we use IPC messaging instead.
func checkNetworkUnavailableFlag() {
#if os(iOS)
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
let isUnavailable = userDefaults?.bool(forKey: GlobalConstants.keyNetworkUnavailable) ?? false
if isUnavailable != networkUnavailable {
AppLogger.shared.log("Network unavailable flag changed: \(isUnavailable)")
networkUnavailable = isUnavailable
updateVPNDisplayState()
}
#endif
// tvOS: Network status is determined by extension state, not a shared flag
}
/// Checks shared app-group container for login required flag set by the network extension.
/// If set, schedules a local notification (if authorized) and shows the authentication UI.
/// iOS only tvOS uses IPC via `checkLoginError` in TVAuthView.
func checkLoginRequiredFlag() {
#if os(iOS)
let userDefaults = UserDefaults(suiteName: GlobalConstants.userPreferencesSuiteName)
guard userDefaults?.bool(forKey: GlobalConstants.keyLoginRequired) == true else { return }
userDefaults?.set(false, forKey: GlobalConstants.keyLoginRequired)
userDefaults?.synchronize()
AppLogger.shared.log("Login required flag detected from extension")
showAuthenticationRequired = true
scheduleLoginRequiredNotification()
#endif
}
#if os(iOS)
private func scheduleLoginRequiredNotification() {
let center = UNUserNotificationCenter.current()
center.getNotificationSettings { settings in
guard settings.authorizationStatus == .authorized else { return }
let content = UNMutableNotificationContent()
content.title = "NetBird"
content.body = "Login required. Please open the app to reconnect."
content.sound = .default
let request = UNNotificationRequest(
identifier: "netbird.login.required",
content: content,
trigger: nil
)
center.add(request) { error in
if let error {
AppLogger.shared.log("Failed to schedule login notification: \(error.localizedDescription)")
}
}
}
}
#endif
}