OSSIPv6Adapter.m 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. //
  2. // OSSIPv6Adapter.m
  3. //
  4. // Created by lingkun on 16/5/16.
  5. // Copyright © 2016 Ali. All rights reserved.
  6. //
  7. #import <Foundation/Foundation.h>
  8. #import "OSSIPv6Adapter.h"
  9. #import "OSSIPv6PrefixResolver.h"
  10. #import "OSSLog.h"
  11. #include <arpa/inet.h>
  12. #include <dns.h>
  13. #include <err.h>
  14. #include <ifaddrs.h>
  15. #include <net/if.h>
  16. #include <netdb.h>
  17. #include <netinet/in.h>
  18. #include <resolv.h>
  19. #include <stdlib.h>
  20. #include <string.h>
  21. #include <sys/socket.h>
  22. #include <sys/sysctl.h>
  23. #if TARGET_OS_IOS
  24. #import <UIKit/UIApplication.h>
  25. #elif TARGET_OS_OSX
  26. #import <AppKit/NSApplication.h>
  27. #endif
  28. #define UNKNOWN_STACK 0
  29. #define SUPPORT_IPV4_STACK 1
  30. #define SUPPORT_IPV6_STACK 2
  31. #define ROUNDUP_LEN(a) \
  32. ((a) > 0 ? (1 + (((a) - 1) | (sizeof(long) - 1))) : sizeof(long))
  33. #define TypeEN "en0"
  34. #define IOS_9_VERSION @"9.0"
  35. @implementation OSSIPv6Adapter
  36. {
  37. BOOL isIPv6Only;
  38. BOOL isIPv6OnlyResolved;
  39. }
  40. - (instancetype)init {
  41. if (self = [super init]) {
  42. isIPv6Only = NO;
  43. isIPv6OnlyResolved = NO;
  44. NSString *notificationName;
  45. #if TARGET_OS_IOS
  46. notificationName = UIApplicationDidBecomeActiveNotification;
  47. #elif TARGET_OS_OSX
  48. notificationName = NSApplicationDidBecomeActiveNotification;
  49. #endif
  50. // When App switches to active status, refresh the IPv6-only check.
  51. NSNotificationCenter *defaultCenter = [NSNotificationCenter defaultCenter];
  52. [defaultCenter addObserver:self
  53. selector:@selector(appDidBecomeActiveFunc)
  54. name:notificationName
  55. object:nil];
  56. }
  57. return self;
  58. }
  59. + (instancetype)getInstance {
  60. static id singletonInstance = nil;
  61. static dispatch_once_t once_token;
  62. dispatch_once(&once_token, ^{
  63. if (!singletonInstance) {
  64. singletonInstance = [[super allocWithZone:NULL] init];
  65. }
  66. });
  67. return singletonInstance;
  68. }
  69. - (BOOL)isIPv6OnlyNetwork {
  70. @synchronized(self) {
  71. if (isIPv6OnlyResolved) {
  72. return isIPv6Only;
  73. }
  74. OSSLogDebug(@"Start resolved network to see if in IPv6-Only env.");
  75. int localStack = 0;
  76. localStack = SUPPORT_IPV4_STACK | SUPPORT_IPV6_STACK;
  77. localStack &= [self getDNSServersIpStack];
  78. if (localStack & SUPPORT_IPV4_STACK) {
  79. // support IPv4
  80. isIPv6Only = NO;
  81. } else if (localStack & SUPPORT_IPV6_STACK) {
  82. // IPv6-Only
  83. isIPv6Only = YES;
  84. [[OSSIPv6PrefixResolver getInstance] updateIPv6Prefix];
  85. } else {
  86. OSSLogDebug(@"[%s]: Error.", __FUNCTION__);
  87. isIPv6Only = NO;
  88. }
  89. isIPv6OnlyResolved = YES;
  90. if (isIPv6Only) {
  91. OSSLogDebug(@"[%s]: IPv6-Only network now.", __FUNCTION__);
  92. } else {
  93. OSSLogDebug(@"[%s]: Not IPv6-Only network now.", __FUNCTION__);
  94. }
  95. return isIPv6Only;
  96. }
  97. }
  98. - (void)appDidBecomeActiveFunc {
  99. OSSLogDebug(@"[%s]: App become active, refresh IPv6-Only status.", __FUNCTION__);
  100. [self reResolveIPv6OnlyStatus];
  101. }
  102. - (BOOL)reResolveIPv6OnlyStatus {
  103. isIPv6OnlyResolved = NO;
  104. return [self isIPv6OnlyNetwork];
  105. }
  106. - (NSString *)handleIpv4Address:(NSString *)addr {
  107. if (addr == nil || addr.length == 0) {
  108. return nil;
  109. }
  110. if ([self isIPv6Address:addr]) return [NSString stringWithFormat:@"[%@]", addr];
  111. NSString *convertedAddr;
  112. if ([self isIPv6OnlyNetwork]) {
  113. convertedAddr = [[OSSIPv6PrefixResolver getInstance] convertIPv4toIPv6:addr];
  114. return [NSString stringWithFormat:@"[%@]", convertedAddr];
  115. } else {
  116. convertedAddr = addr;
  117. }
  118. return convertedAddr;
  119. }
  120. /**
  121. * @brief Looks up the DNS server stack and returns the flag combinations of SUPPORT_IPV4_STACK and SUPPORT_IPV6_STACK.
  122. *
  123. * @return the flag combinations of SUPPORT_IPV4_STACK and SUPPORT_IPV6_STACK
  124. */
  125. - (int)getDNSServersIpStack {
  126. int dns_stack = 0;
  127. res_state res = malloc(sizeof(struct __res_state));
  128. int result = res_ninit(res);
  129. if (result == 0) {
  130. union res_9_sockaddr_union *addr_union = malloc(res->nscount * sizeof(union res_9_sockaddr_union));
  131. res_getservers(res, addr_union, res->nscount);
  132. for (int i = 0; i < res->nscount; i++) {
  133. if (addr_union[i].sin.sin_family == AF_INET) {
  134. char ip[INET_ADDRSTRLEN];
  135. if (inet_ntop(AF_INET, &(addr_union[i].sin.sin_addr), ip, INET_ADDRSTRLEN)) {
  136. dns_stack |= SUPPORT_IPV4_STACK;
  137. }
  138. } else if (addr_union[i].sin6.sin6_family == AF_INET6) {
  139. char ip[INET6_ADDRSTRLEN];
  140. if (inet_ntop(AF_INET6, &(addr_union[i].sin6.sin6_addr), ip, INET6_ADDRSTRLEN)) {
  141. dns_stack |= SUPPORT_IPV6_STACK;
  142. }
  143. } else {
  144. OSSLogDebug(@"%s: Undefined family.", __FUNCTION__);
  145. }
  146. }
  147. free(addr_union);
  148. }
  149. res_ndestroy(res);
  150. free(res);
  151. return dns_stack;
  152. }
  153. - (BOOL)isIPv4Address:(NSString *)addr {
  154. if (addr == nil) {
  155. return NO;
  156. }
  157. const char *utf8 = [addr UTF8String];
  158. // Check valid IPv4.
  159. struct in_addr dst;
  160. int success = inet_pton(AF_INET, utf8, &(dst.s_addr));
  161. return success == 1;
  162. }
  163. - (BOOL)isIPv6Address:(NSString *)addr {
  164. if (addr == nil) {
  165. return NO;
  166. }
  167. const char *utf8 = [addr UTF8String];
  168. // Check valid IPv6.
  169. struct in6_addr dst6;
  170. int success = inet_pton(AF_INET6, utf8, &dst6);
  171. return (success == 1);
  172. }
  173. @end