import Foundation import Network public enum LoopbackHost { public static func isLoopback(_ rawHost: String) -> Bool { self.isLoopbackHost(rawHost) } public static func isLoopbackHost(_ rawHost: String) -> Bool { var host = rawHost .trimmingCharacters(in: .whitespacesAndNewlines) .lowercased() .trimmingCharacters(in: CharacterSet(charactersIn: "[]")) if host.hasSuffix(".") { host.removeLast() } if let zoneIndex = host.firstIndex(of: "%") { host = String(host[.. Bool { let host = rawHost.trimmingCharacters(in: .whitespacesAndNewlines).lowercased() guard !host.isEmpty else { return false } if self.isLoopbackHost(host) { return true } if host.hasSuffix(".local") { return true } if host.hasSuffix(".ts.net") { return true } if host.hasSuffix(".tailscale.net") { return true } // Allow MagicDNS / LAN hostnames like "peters-mac-studio-1". if !host.contains("."), !host.contains(":") { return true } guard let ipv4 = self.parseIPv4(host) else { return false } return self.isLocalNetworkIPv4(ipv4) } static func parseIPv4(_ host: String) -> (UInt8, UInt8, UInt8, UInt8)? { let parts = host.split(separator: ".", omittingEmptySubsequences: false) guard parts.count == 4 else { return nil } let bytes: [UInt8] = parts.compactMap { UInt8($0) } guard bytes.count == 4 else { return nil } return (bytes[0], bytes[1], bytes[2], bytes[3]) } static func isLocalNetworkIPv4(_ ip: (UInt8, UInt8, UInt8, UInt8)) -> Bool { let (a, b, _, _) = ip // 10.0.0.0/8 if a == 10 { return true } // 172.16.0.0/12 if a == 172, (16...31).contains(Int(b)) { return true } // 192.168.0.0/16 if a == 192, b == 168 { return true } // 127.0.0.0/8 if a == 127 { return true } // 169.254.0.0/16 (link-local) if a == 169, b == 254 { return true } // Tailscale: 100.64.0.0/10 if a == 100, (64...127).contains(Int(b)) { return true } return false } }