Networkstack - let OS decide IPv4 or IPv6
This commit is contained in:
		
							parent
							
								
									2ead9046e9
								
							
						
					
					
						commit
						5f673eadcb
					
				|  | @ -154,7 +154,6 @@ public class DatabaseBackend extends SQLiteOpenHelper { | ||||||
| 	private static String CREATE_RESOLVER_RESULTS_TABLE = "create table " + RESOLVER_RESULTS_TABLENAME + "(" | 	private static String CREATE_RESOLVER_RESULTS_TABLE = "create table " + RESOLVER_RESULTS_TABLENAME + "(" | ||||||
| 			+ Resolver.Result.DOMAIN + " TEXT," | 			+ Resolver.Result.DOMAIN + " TEXT," | ||||||
| 			+ Resolver.Result.HOSTNAME + " TEXT," | 			+ Resolver.Result.HOSTNAME + " TEXT," | ||||||
| 			+ Resolver.Result.IP + " BLOB," |  | ||||||
| 			+ Resolver.Result.PRIORITY + " NUMBER," | 			+ Resolver.Result.PRIORITY + " NUMBER," | ||||||
| 			+ Resolver.Result.DIRECT_TLS + " NUMBER," | 			+ Resolver.Result.DIRECT_TLS + " NUMBER," | ||||||
| 			+ Resolver.Result.AUTHENTICATED + " NUMBER," | 			+ Resolver.Result.AUTHENTICATED + " NUMBER," | ||||||
|  |  | ||||||
|  | @ -7,8 +7,6 @@ import android.util.Log; | ||||||
| 
 | 
 | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.lang.reflect.Field; | import java.lang.reflect.Field; | ||||||
| import java.net.Inet4Address; |  | ||||||
| import java.net.InetAddress; |  | ||||||
| import java.net.UnknownHostException; | import java.net.UnknownHostException; | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
|  | @ -25,11 +23,8 @@ import de.measite.minidns.hla.DnssecResolverApi; | ||||||
| import de.measite.minidns.hla.ResolverApi; | import de.measite.minidns.hla.ResolverApi; | ||||||
| import de.measite.minidns.hla.ResolverResult; | import de.measite.minidns.hla.ResolverResult; | ||||||
| import de.measite.minidns.iterative.ReliableDNSClient; | import de.measite.minidns.iterative.ReliableDNSClient; | ||||||
| import de.measite.minidns.record.A; |  | ||||||
| import de.measite.minidns.record.AAAA; |  | ||||||
| import de.measite.minidns.record.CNAME; | import de.measite.minidns.record.CNAME; | ||||||
| import de.measite.minidns.record.Data; | import de.measite.minidns.record.Data; | ||||||
| import de.measite.minidns.record.InternetAddressRR; |  | ||||||
| import de.measite.minidns.record.SRV; | import de.measite.minidns.record.SRV; | ||||||
| import eu.siacs.conversations.Config; | import eu.siacs.conversations.Config; | ||||||
| import eu.siacs.conversations.R; | import eu.siacs.conversations.R; | ||||||
|  | @ -37,10 +32,10 @@ import eu.siacs.conversations.services.XmppConnectionService; | ||||||
| 
 | 
 | ||||||
| public class Resolver { | public class Resolver { | ||||||
| 
 | 
 | ||||||
|     public static final int DEFAULT_PORT_XMPP = 5222; |     public static final int DEFAULT_PORT_XMPP = 5223; | ||||||
| 
 | 
 | ||||||
|     private static final String DIRECT_TLS_SERVICE = "_xmpps-client"; |     private static final String DIRECT_TLS_SERVICE = "_xmpps-client"; | ||||||
|     private static final String STARTTLS_SERICE = "_xmpp-client"; |     private static final String STARTTLS_SERVICE = "_xmpp-client"; | ||||||
| 
 | 
 | ||||||
|     private static XmppConnectionService SERVICE = null; |     private static XmppConnectionService SERVICE = null; | ||||||
| 
 | 
 | ||||||
|  | @ -65,7 +60,9 @@ public class Resolver { | ||||||
|             final Field useHardcodedDnsServers = DNSClient.class.getDeclaredField("useHardcodedDnsServers"); |             final Field useHardcodedDnsServers = DNSClient.class.getDeclaredField("useHardcodedDnsServers"); | ||||||
|             useHardcodedDnsServers.setAccessible(true); |             useHardcodedDnsServers.setAccessible(true); | ||||||
|             useHardcodedDnsServers.setBoolean(dnsClient, false); |             useHardcodedDnsServers.setBoolean(dnsClient, false); | ||||||
|         } catch (NoSuchFieldException | IllegalAccessException e) { |         } catch (NoSuchFieldException e) { | ||||||
|  |             Log.e(Config.LOGTAG, "Unable to disable hardcoded DNS servers", e); | ||||||
|  |         } catch (IllegalAccessException e) { | ||||||
|             Log.e(Config.LOGTAG, "Unable to disable hardcoded DNS servers", e); |             Log.e(Config.LOGTAG, "Unable to disable hardcoded DNS servers", e); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -81,7 +78,7 @@ public class Resolver { | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|     public static List<Result> resolve(String domain) { |     public static List<Result> resolve(String domain) { | ||||||
|         final  List<Result> ipResults = fromIpAddress(domain); |         final List<Result> ipResults = fromIpAddress(domain); | ||||||
|         if (ipResults.size() > 0) { |         if (ipResults.size() > 0) { | ||||||
|             return ipResults; |             return ipResults; | ||||||
|         } |         } | ||||||
|  | @ -147,83 +144,47 @@ public class Resolver { | ||||||
|         if (!IP.matches(domain)) { |         if (!IP.matches(domain)) { | ||||||
|             return Collections.emptyList(); |             return Collections.emptyList(); | ||||||
|         } |         } | ||||||
|         try { |         final List<Result> results = new ArrayList<>(); | ||||||
|             Result result = new Result(); |         DNSName hostname = DNSName.from(domain); | ||||||
|             result.ip = InetAddress.getByName(domain); |         results.add(Result.createDefault(hostname)); | ||||||
|             result.port = DEFAULT_PORT_XMPP; | 
 | ||||||
|             return Collections.singletonList(result); |         Result result = new Result(); | ||||||
|         } catch (UnknownHostException e) { |         result.hostname = hostname; | ||||||
|             return Collections.emptyList(); |         result.port = 5222; | ||||||
|         } |         results.add(result); | ||||||
|  | 
 | ||||||
|  |         return results; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private static List<Result> resolveSrv(String domain, final boolean directTls) throws IOException { |     private static List<Result> resolveSrv(String domain, final boolean directTls) throws IOException { | ||||||
|         DNSName dnsName = DNSName.from((directTls ? DIRECT_TLS_SERVICE : STARTTLS_SERICE) + "._tcp." + domain); |         DNSName dnsName = DNSName.from((directTls ? DIRECT_TLS_SERVICE : STARTTLS_SERVICE) + "._tcp." + domain); | ||||||
|         ResolverResult<SRV> result = resolveWithFallback(dnsName, SRV.class); |         ResolverResult<SRV> result = resolveWithFallback(dnsName, SRV.class); | ||||||
|         final List<Result> results = new ArrayList<>(); |         final List<Result> results = new ArrayList<>(); | ||||||
|         final List<Thread> threads = new ArrayList<>(); |  | ||||||
|         for (SRV record : result.getAnswersOrEmptySet()) { |         for (SRV record : result.getAnswersOrEmptySet()) { | ||||||
|             if (record.name.length() == 0 && record.priority == 0) { |             if (record.name.length() == 0 && record.priority == 0) { | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|             threads.add(new Thread(() -> { |             results.add(Result.fromRecord(record, directTls)); | ||||||
|                 final List<Result> ipv4s = resolveIp(record, A.class, result.isAuthenticData(), directTls); |         } | ||||||
|                 if (ipv4s.size() == 0) { |  | ||||||
|                     Result resolverResult = Result.fromRecord(record, directTls); |  | ||||||
|                     resolverResult.authenticated = resolverResult.isAuthenticated(); |  | ||||||
|                     ipv4s.add(resolverResult); |  | ||||||
|                 } |  | ||||||
|                 synchronized (results) { |  | ||||||
|                     results.addAll(ipv4s); |  | ||||||
|                 } |  | ||||||
| 
 | 
 | ||||||
|             })); |         if (results.size() == 0) { | ||||||
|             threads.add(new Thread(() -> { |             for (SRV record : result.getAnswersOrEmptySet()) { | ||||||
|                 final List<Result> ipv6s = resolveIp(record, AAAA.class, result.isAuthenticData(), directTls); |                 if (record.name.length() == 0 && record.priority == 0) { | ||||||
|                 synchronized (results) { |                     continue; | ||||||
|                     results.addAll(ipv6s); |  | ||||||
|                 } |                 } | ||||||
|             })); |                 Result resolverResult = Result.fromRecord(record, directTls); | ||||||
|         } |                 resolverResult.authenticated = resolverResult.isAuthenticated(); | ||||||
|         for (Thread thread : threads) { |                 results.add(resolverResult); | ||||||
|             thread.start(); |  | ||||||
|         } |  | ||||||
|         for (Thread thread : threads) { |  | ||||||
|             try { |  | ||||||
|                 thread.join(); |  | ||||||
|             } catch (InterruptedException e) { |  | ||||||
|                 return Collections.emptyList(); |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|         return results; |         return results; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private static <D extends InternetAddressRR> List<Result> resolveIp(SRV srv, Class<D> type, boolean authenticated, boolean directTls) { |  | ||||||
|         List<Result> list = new ArrayList<>(); |  | ||||||
|         try { |  | ||||||
|             ResolverResult<D> results = resolveWithFallback(srv.name, type, authenticated); |  | ||||||
|             for (D record : results.getAnswersOrEmptySet()) { |  | ||||||
|                 Result resolverResult = Result.fromRecord(srv, directTls); |  | ||||||
|                 resolverResult.authenticated = results.isAuthenticData() && authenticated; |  | ||||||
|                 resolverResult.ip = record.getInetAddress(); |  | ||||||
|                 list.add(resolverResult); |  | ||||||
|             } |  | ||||||
|         } catch (Throwable t) { |  | ||||||
|             Log.d(Config.LOGTAG, Resolver.class.getSimpleName() + ": error resolving " + type.getSimpleName() + " " + t.getMessage()); |  | ||||||
|         } |  | ||||||
|         return list; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private static List<Result> resolveNoSrvRecords(DNSName dnsName, boolean withCnames) { |     private static List<Result> resolveNoSrvRecords(DNSName dnsName, boolean withCnames) { | ||||||
|         List<Result> results = new ArrayList<>(); |         List<Result> results = new ArrayList<>(); | ||||||
|         try { |         try { | ||||||
|             for (A a : resolveWithFallback(dnsName, A.class, false).getAnswersOrEmptySet()) { |             if (withCnames) { | ||||||
|                 results.add(Result.createDefault(dnsName, a.getInetAddress())); |  | ||||||
|             } |  | ||||||
|             for (AAAA aaaa : resolveWithFallback(dnsName, AAAA.class, false).getAnswersOrEmptySet()) { |  | ||||||
|                 results.add(Result.createDefault(dnsName, aaaa.getInetAddress())); |  | ||||||
|             } |  | ||||||
|             if (results.size() == 0 && withCnames) { |  | ||||||
|                 for (CNAME cname : resolveWithFallback(dnsName, CNAME.class, false).getAnswersOrEmptySet()) { |                 for (CNAME cname : resolveWithFallback(dnsName, CNAME.class, false).getAnswersOrEmptySet()) { | ||||||
|                     results.addAll(resolveNoSrvRecords(cname.name, false)); |                     results.addAll(resolveNoSrvRecords(cname.name, false)); | ||||||
|                 } |                 } | ||||||
|  | @ -232,6 +193,12 @@ public class Resolver { | ||||||
|             Log.d(Config.LOGTAG, Resolver.class.getSimpleName() + "error resolving fallback records", throwable); |             Log.d(Config.LOGTAG, Resolver.class.getSimpleName() + "error resolving fallback records", throwable); | ||||||
|         } |         } | ||||||
|         results.add(Result.createDefault(dnsName)); |         results.add(Result.createDefault(dnsName)); | ||||||
|  | 
 | ||||||
|  |         Result result = new Result(); | ||||||
|  |         result.hostname = dnsName; | ||||||
|  |         result.port = 5222; | ||||||
|  |         results.add(result); | ||||||
|  | 
 | ||||||
|         return results; |         return results; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -262,18 +229,17 @@ public class Resolver { | ||||||
| 
 | 
 | ||||||
|     public static class Result implements Comparable<Result> { |     public static class Result implements Comparable<Result> { | ||||||
|         public static final String DOMAIN = "domain"; |         public static final String DOMAIN = "domain"; | ||||||
|         public static final String IP = "ip"; |  | ||||||
|         public static final String HOSTNAME = "hostname"; |         public static final String HOSTNAME = "hostname"; | ||||||
|         public static final String PORT = "port"; |         public static final String PORT = "port"; | ||||||
|         public static final String PRIORITY = "priority"; |         public static final String PRIORITY = "priority"; | ||||||
|         public static final String DIRECT_TLS = "directTls"; |         public static final String DIRECT_TLS = "directTls"; | ||||||
|         public static final String AUTHENTICATED = "authenticated"; |         public static final String AUTHENTICATED = "authenticated"; | ||||||
|         private InetAddress ip; |  | ||||||
|         private DNSName hostname; |         private DNSName hostname; | ||||||
|         private int port = DEFAULT_PORT_XMPP; |         private int port = DEFAULT_PORT_XMPP; | ||||||
|         private boolean directTls = false; |         private boolean directTls = true; | ||||||
|         private boolean authenticated = false; |         private boolean authenticated = false; | ||||||
|         private int priority; |         private int priority; | ||||||
|  |         private long rtt = -1; | ||||||
| 
 | 
 | ||||||
|         static Result fromRecord(SRV srv, boolean directTls) { |         static Result fromRecord(SRV srv, boolean directTls) { | ||||||
|             Result result = new Result(); |             Result result = new Result(); | ||||||
|  | @ -284,25 +250,15 @@ public class Resolver { | ||||||
|             return result; |             return result; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         static Result createDefault(DNSName hostname, InetAddress ip) { |         static Result createDefault(DNSName hostname) { | ||||||
|             Result result = new Result(); |             Result result = new Result(); | ||||||
|             result.port = DEFAULT_PORT_XMPP; |             result.port = DEFAULT_PORT_XMPP; | ||||||
|             result.hostname = hostname; |             result.hostname = hostname; | ||||||
|             result.ip = ip; |  | ||||||
|             return result; |             return result; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         static Result createDefault(DNSName hostname) { |  | ||||||
|             return createDefault(hostname, null); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static Result fromCursor(Cursor cursor) { |         public static Result fromCursor(Cursor cursor) { | ||||||
|             final Result result = new Result(); |             final Result result = new Result(); | ||||||
|             try { |  | ||||||
|                 result.ip = InetAddress.getByAddress(cursor.getBlob(cursor.getColumnIndex(IP))); |  | ||||||
|             } catch (UnknownHostException e) { |  | ||||||
|                 result.ip = null; |  | ||||||
|             } |  | ||||||
|             final String hostname = cursor.getString(cursor.getColumnIndex(HOSTNAME)); |             final String hostname = cursor.getString(cursor.getColumnIndex(HOSTNAME)); | ||||||
|             result.hostname = hostname == null ? null : DNSName.from(hostname); |             result.hostname = hostname == null ? null : DNSName.from(hostname); | ||||||
|             result.port = cursor.getInt(cursor.getColumnIndex(PORT)); |             result.port = cursor.getInt(cursor.getColumnIndex(PORT)); | ||||||
|  | @ -323,14 +279,12 @@ public class Resolver { | ||||||
|             if (directTls != result.directTls) return false; |             if (directTls != result.directTls) return false; | ||||||
|             if (authenticated != result.authenticated) return false; |             if (authenticated != result.authenticated) return false; | ||||||
|             if (priority != result.priority) return false; |             if (priority != result.priority) return false; | ||||||
|             if (ip != null ? !ip.equals(result.ip) : result.ip != null) return false; |  | ||||||
|             return hostname != null ? hostname.equals(result.hostname) : result.hostname == null; |             return hostname != null ? hostname.equals(result.hostname) : result.hostname == null; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         @Override |         @Override | ||||||
|         public int hashCode() { |         public int hashCode() { | ||||||
|             int result = ip != null ? ip.hashCode() : 0; |             int result = hostname != null ? hostname.hashCode() : 0; | ||||||
|             result = 31 * result + (hostname != null ? hostname.hashCode() : 0); |  | ||||||
|             result = 31 * result + port; |             result = 31 * result + port; | ||||||
|             result = 31 * result + (directTls ? 1 : 0); |             result = 31 * result + (directTls ? 1 : 0); | ||||||
|             result = 31 * result + (authenticated ? 1 : 0); |             result = 31 * result + (authenticated ? 1 : 0); | ||||||
|  | @ -338,10 +292,6 @@ public class Resolver { | ||||||
|             return result; |             return result; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public InetAddress getIp() { |  | ||||||
|             return ip; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public int getPort() { |         public int getPort() { | ||||||
|             return port; |             return port; | ||||||
|         } |         } | ||||||
|  | @ -361,7 +311,6 @@ public class Resolver { | ||||||
|         @Override |         @Override | ||||||
|         public String toString() { |         public String toString() { | ||||||
|             return "Result{" + |             return "Result{" + | ||||||
|                     "ip='" + (ip == null ? null : ip.getHostAddress()) + '\'' + |  | ||||||
|                     ", hostame='" + hostname.toString() + '\'' + |                     ", hostame='" + hostname.toString() + '\'' + | ||||||
|                     ", port=" + port + |                     ", port=" + port + | ||||||
|                     ", directTls=" + directTls + |                     ", directTls=" + directTls + | ||||||
|  | @ -374,17 +323,7 @@ public class Resolver { | ||||||
|         public int compareTo(@NonNull Result result) { |         public int compareTo(@NonNull Result result) { | ||||||
|             if (result.priority == priority) { |             if (result.priority == priority) { | ||||||
|                 if (directTls == result.directTls) { |                 if (directTls == result.directTls) { | ||||||
|                     if (ip == null && result.ip == null) { |  | ||||||
|                         return 0; |                         return 0; | ||||||
|                     } else if (ip != null && result.ip != null) { |  | ||||||
|                         if (ip instanceof Inet4Address && result.ip instanceof Inet4Address) { |  | ||||||
|                             return 0; |  | ||||||
|                         } else { |  | ||||||
|                             return ip instanceof Inet4Address ? -1 : 1; |  | ||||||
|                         } |  | ||||||
|                     } else { |  | ||||||
|                         return ip != null ? -1 : 1; |  | ||||||
|                     } |  | ||||||
|                 } else { |                 } else { | ||||||
|                     return directTls ? -1 : 1; |                     return directTls ? -1 : 1; | ||||||
|                 } |                 } | ||||||
|  | @ -395,7 +334,6 @@ public class Resolver { | ||||||
| 
 | 
 | ||||||
|         public ContentValues toContentValues() { |         public ContentValues toContentValues() { | ||||||
|             final ContentValues contentValues = new ContentValues(); |             final ContentValues contentValues = new ContentValues(); | ||||||
|             contentValues.put(IP, ip == null ? null : ip.getAddress()); |  | ||||||
|             contentValues.put(HOSTNAME, hostname == null ? null : hostname.toString()); |             contentValues.put(HOSTNAME, hostname == null ? null : hostname.toString()); | ||||||
|             contentValues.put(PORT, port); |             contentValues.put(PORT, port); | ||||||
|             contentValues.put(PRIORITY, priority); |             contentValues.put(PRIORITY, priority); | ||||||
|  |  | ||||||
|  | @ -75,7 +75,6 @@ import eu.siacs.conversations.services.MessageArchiveService; | ||||||
| import eu.siacs.conversations.services.NotificationService; | import eu.siacs.conversations.services.NotificationService; | ||||||
| import eu.siacs.conversations.services.XmppConnectionService; | import eu.siacs.conversations.services.XmppConnectionService; | ||||||
| import eu.siacs.conversations.utils.CryptoHelper; | import eu.siacs.conversations.utils.CryptoHelper; | ||||||
| import eu.siacs.conversations.utils.IP; |  | ||||||
| import eu.siacs.conversations.utils.Patterns; | import eu.siacs.conversations.utils.Patterns; | ||||||
| import eu.siacs.conversations.utils.Resolver; | import eu.siacs.conversations.utils.Resolver; | ||||||
| import eu.siacs.conversations.utils.SSLSocketHelper; | import eu.siacs.conversations.utils.SSLSocketHelper; | ||||||
|  | @ -315,18 +314,10 @@ public class XmppConnection implements Runnable { | ||||||
|                         features.encryptionEnabled = result.isDirectTls(); |                         features.encryptionEnabled = result.isDirectTls(); | ||||||
|                         verifiedHostname = result.isAuthenticated() ? result.getHostname().toString() : null; |                         verifiedHostname = result.isAuthenticated() ? result.getHostname().toString() : null; | ||||||
|                         Log.d(Config.LOGTAG,"verified hostname "+verifiedHostname); |                         Log.d(Config.LOGTAG,"verified hostname "+verifiedHostname); | ||||||
|                         final InetSocketAddress addr; |                         final InetSocketAddress addr = new InetSocketAddress(IDN.toASCII(result.getHostname().toString()), result.getPort()); | ||||||
|                         if (result.getIp() != null) { |                         Log.d(Config.LOGTAG, account.getJid().asBareJid().toString() | ||||||
|                             addr = new InetSocketAddress(result.getIp(), result.getPort()); |                                 + ": using values from resolver " | ||||||
|                             Log.d(Config.LOGTAG, account.getJid().asBareJid().toString() |                                 + result.getHostname().toString() + ":" + result.getPort() + " tls: " + features.encryptionEnabled); | ||||||
|                                     + ": using values from resolver " + (result.getHostname() == null ? "" : result.getHostname().toString() |  | ||||||
|                                     + "/") + result.getIp().getHostAddress() + ":" + result.getPort() + " tls: " + features.encryptionEnabled); |  | ||||||
|                         } else { |  | ||||||
|                             addr = new InetSocketAddress(IDN.toASCII(result.getHostname().toString()), result.getPort()); |  | ||||||
|                             Log.d(Config.LOGTAG, account.getJid().asBareJid().toString() |  | ||||||
|                                     + ": using values from resolver " |  | ||||||
|                                     + result.getHostname().toString() + ":" + result.getPort() + " tls: " + features.encryptionEnabled); |  | ||||||
|                         } |  | ||||||
| 
 | 
 | ||||||
|                         if (!features.encryptionEnabled) { |                         if (!features.encryptionEnabled) { | ||||||
|                             localSocket = new Socket(); |                             localSocket = new Socket(); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue