196 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			196 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System.Diagnostics.CodeAnalysis;
 | |
| using System.Net;
 | |
| using CoreFoundation;
 | |
| using CoreTelephony;
 | |
| using SystemConfiguration;
 | |
| 
 | |
| namespace Blahblah.Library.Network;
 | |
| 
 | |
| [SuppressMessage("Interoperability", "CA1416:Validate platform compatibility", Justification = "<Pending>")]
 | |
| partial class Connectivity
 | |
| {
 | |
|     static readonly Lazy<CTCellularData> cellularData = new(() => new CTCellularData());
 | |
| 
 | |
|     static ReachabilityListener? listener;
 | |
| 
 | |
|     private static partial NetworkStatus PlatformGetStatus()
 | |
|     {
 | |
|         var restricted = cellularData.Value.RestrictedState == CTCellularDataRestrictedState.Restricted;
 | |
| 
 | |
|         var internetStatus = InternetConnectionStatus();
 | |
|         if (internetStatus == NetworkStatus.ReachableViaCarrierDataNetwork && !restricted)
 | |
|         {
 | |
|             return NetworkStatus.ReachableViaCarrierDataNetwork;
 | |
|         }
 | |
|         if (internetStatus == NetworkStatus.ReachableViaWiFiNetwork)
 | |
|         {
 | |
|             return NetworkStatus.ReachableViaWiFiNetwork;
 | |
|         }
 | |
| 
 | |
|         var remoteStatus = RemoteHostStatus();
 | |
|         if (remoteStatus == NetworkStatus.ReachableViaCarrierDataNetwork && !restricted)
 | |
|         {
 | |
|             return NetworkStatus.ReachableViaCarrierDataNetwork;
 | |
|         }
 | |
|         if (remoteStatus == NetworkStatus.ReachableViaWiFiNetwork)
 | |
|         {
 | |
|             return NetworkStatus.ReachableViaWiFiNetwork;
 | |
|         }
 | |
| 
 | |
|         return NetworkStatus.NotReachable;
 | |
|     }
 | |
| 
 | |
|     static NetworkStatus RemoteHostStatus()
 | |
|     {
 | |
|         using var remote = new NetworkReachability(hostName);
 | |
| 
 | |
|         if (!remote.TryGetFlags(out var flags))
 | |
|         {
 | |
|             return NetworkStatus.NotReachable;
 | |
|         }
 | |
| 
 | |
|         if (!IsReachableWithoutRequiringConnection(flags))
 | |
|         {
 | |
|             return NetworkStatus.NotReachable;
 | |
|         }
 | |
| 
 | |
|         if ((flags & NetworkReachabilityFlags.IsWWAN) != 0)
 | |
|         {
 | |
|             return NetworkStatus.ReachableViaCarrierDataNetwork;
 | |
|         }
 | |
| 
 | |
|         return NetworkStatus.ReachableViaWiFiNetwork;
 | |
| 
 | |
|     }
 | |
| 
 | |
|     static bool IsReachableWithoutRequiringConnection(NetworkReachabilityFlags flags)
 | |
|     {
 | |
|         // Is it reachable with the current network configuration?
 | |
|         var isReachable = (flags & NetworkReachabilityFlags.Reachable) != 0;
 | |
| 
 | |
|         // Do we need a connection to reach it?
 | |
|         var noConnectionRequired = (flags & NetworkReachabilityFlags.ConnectionRequired) == 0;
 | |
| 
 | |
|         // Since the network stack will automatically try to get the WAN up,
 | |
|         // probe that
 | |
|         if ((flags & NetworkReachabilityFlags.IsWWAN) != 0)
 | |
|         {
 | |
|             noConnectionRequired = true;
 | |
|         }
 | |
| 
 | |
|         return isReachable && noConnectionRequired;
 | |
|     }
 | |
| 
 | |
|     static NetworkStatus InternetConnectionStatus()
 | |
|     {
 | |
|         var status = NetworkStatus.NotReachable;
 | |
| 
 | |
|         var defaultNetworkAvailable = IsNetworkAvailable(out var flags);
 | |
| 
 | |
|         // If the connection is reachable and no connection is required, then assume it's WiFi
 | |
|         if (defaultNetworkAvailable)
 | |
|         {
 | |
|             status = NetworkStatus.ReachableViaWiFiNetwork;
 | |
|         }
 | |
|         else if ((flags & NetworkReachabilityFlags.IsWWAN) != 0)
 | |
|         {
 | |
|             // If it's a WWAN connection..
 | |
|             status = NetworkStatus.ReachableViaCarrierDataNetwork;
 | |
|         }
 | |
| 
 | |
|         // If the connection is on-demand or on-traffic and no user intervention
 | |
|         // is required, then assume WiFi.
 | |
|         if (((flags & NetworkReachabilityFlags.ConnectionOnDemand) != 0 || (flags & NetworkReachabilityFlags.ConnectionOnTraffic) != 0) &&
 | |
|              (flags & NetworkReachabilityFlags.InterventionRequired) == 0)
 | |
|         {
 | |
|             status = NetworkStatus.ReachableViaWiFiNetwork;
 | |
|         }
 | |
| 
 | |
|         return status;
 | |
|     }
 | |
| 
 | |
|     private static bool IsNetworkAvailable(out NetworkReachabilityFlags flags)
 | |
|     {
 | |
|         var ip = new IPAddress(0);
 | |
|         using var route = new NetworkReachability(ip);
 | |
| 
 | |
|         if (!route.TryGetFlags(out flags))
 | |
|         {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         return IsReachableWithoutRequiringConnection(flags);
 | |
|     }
 | |
| 
 | |
|     private static partial void StartListeners()
 | |
|     {
 | |
|         StopListeners();
 | |
| 
 | |
|         listener = new ReachabilityListener();
 | |
|         listener.ReachabilityChanged += OnConnectivityChanged;
 | |
|     }
 | |
| 
 | |
|     private static partial void StopListeners()
 | |
|     {
 | |
|         if (listener == null)
 | |
|         {
 | |
|             return;
 | |
|         }
 | |
|         listener.ReachabilityChanged -= OnConnectivityChanged;
 | |
|         listener.Dispose();
 | |
|         listener = null;
 | |
|     }
 | |
| 
 | |
|     class ReachabilityListener : IDisposable
 | |
|     {
 | |
|         NetworkReachability? routeReachability;
 | |
|         NetworkReachability? remoteReachability;
 | |
| 
 | |
|         public event Action? ReachabilityChanged;
 | |
| 
 | |
|         public ReachabilityListener()
 | |
|         {
 | |
|             var ip = new IPAddress(0);
 | |
|             routeReachability = new NetworkReachability(ip);
 | |
|             routeReachability.SetNotification(OnChange);
 | |
|             routeReachability.Schedule(CFRunLoop.Main, CFRunLoop.ModeDefault);
 | |
| 
 | |
|             remoteReachability = new NetworkReachability(hostName);
 | |
| 
 | |
|             // Need to probe before we queue, or we wont get any meaningful values
 | |
|             // this only happens when you create NetworkReachability from a hostname
 | |
|             remoteReachability.TryGetFlags(out var flags);
 | |
| 
 | |
|             remoteReachability.SetNotification(OnChange);
 | |
|             remoteReachability.Schedule(CFRunLoop.Main, CFRunLoop.ModeDefault);
 | |
| 
 | |
|             cellularData.Value.RestrictionDidUpdateNotifier = new(OnRestrictedStateChanged);
 | |
|         }
 | |
| 
 | |
|         public void Dispose()
 | |
|         {
 | |
|             routeReachability?.Dispose();
 | |
|             routeReachability = null;
 | |
|             remoteReachability?.Dispose();
 | |
|             remoteReachability = null;
 | |
| 
 | |
|             cellularData.Value.RestrictionDidUpdateNotifier = null;
 | |
|         }
 | |
| 
 | |
|         void OnRestrictedStateChanged(CTCellularDataRestrictedState state)
 | |
|         {
 | |
|             ReachabilityChanged?.Invoke();
 | |
|         }
 | |
| 
 | |
|         async void OnChange(NetworkReachabilityFlags flags)
 | |
|         {
 | |
|             // Add in artificial delay so the connection status has time to change
 | |
|             // else it will return true no matter what.
 | |
|             await Task.Delay(100);
 | |
| 
 | |
|             ReachabilityChanged?.Invoke();
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 |