Networkstack - let OS decide IPv4 or IPv6
This commit is contained in:
		
							parent
							
								
									18bcc0b2eb
								
							
						
					
					
						commit
						196b351fd3
					
				|  | @ -154,7 +154,6 @@ public class DatabaseBackend extends SQLiteOpenHelper { | |||
| 	private static String CREATE_RESOLVER_RESULTS_TABLE = "create table " + RESOLVER_RESULTS_TABLENAME + "(" | ||||
| 			+ Resolver.Result.DOMAIN + " TEXT," | ||||
| 			+ Resolver.Result.HOSTNAME + " TEXT," | ||||
| 			+ Resolver.Result.IP + " BLOB," | ||||
| 			+ Resolver.Result.PRIORITY + " NUMBER," | ||||
| 			+ Resolver.Result.DIRECT_TLS + " NUMBER," | ||||
| 			+ Resolver.Result.AUTHENTICATED + " NUMBER," | ||||
|  |  | |||
|  | @ -7,8 +7,6 @@ import android.util.Log; | |||
| 
 | ||||
| import java.io.IOException; | ||||
| import java.lang.reflect.Field; | ||||
| import java.net.Inet4Address; | ||||
| import java.net.InetAddress; | ||||
| import java.net.UnknownHostException; | ||||
| import java.util.ArrayList; | ||||
| 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.ResolverResult; | ||||
| 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.Data; | ||||
| import de.measite.minidns.record.InternetAddressRR; | ||||
| import de.measite.minidns.record.SRV; | ||||
| import eu.siacs.conversations.Config; | ||||
| import eu.siacs.conversations.R; | ||||
|  | @ -37,10 +32,10 @@ import eu.siacs.conversations.services.XmppConnectionService; | |||
| 
 | ||||
| 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 STARTTLS_SERICE = "_xmpp-client"; | ||||
|     private static final String STARTTLS_SERVICE = "_xmpp-client"; | ||||
| 
 | ||||
|     private static XmppConnectionService SERVICE = null; | ||||
| 
 | ||||
|  | @ -65,7 +60,9 @@ public class Resolver { | |||
|             final Field useHardcodedDnsServers = DNSClient.class.getDeclaredField("useHardcodedDnsServers"); | ||||
|             useHardcodedDnsServers.setAccessible(true); | ||||
|             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); | ||||
|         } | ||||
|     } | ||||
|  | @ -147,83 +144,47 @@ public class Resolver { | |||
|         if (!IP.matches(domain)) { | ||||
|             return Collections.emptyList(); | ||||
|         } | ||||
|         try { | ||||
|         final List<Result> results = new ArrayList<>(); | ||||
|         DNSName hostname = DNSName.from(domain); | ||||
|         results.add(Result.createDefault(hostname)); | ||||
| 
 | ||||
|         Result result = new Result(); | ||||
|             result.ip = InetAddress.getByName(domain); | ||||
|             result.port = DEFAULT_PORT_XMPP; | ||||
|             return Collections.singletonList(result); | ||||
|         } catch (UnknownHostException e) { | ||||
|             return Collections.emptyList(); | ||||
|         } | ||||
|         result.hostname = hostname; | ||||
|         result.port = 5222; | ||||
|         results.add(result); | ||||
| 
 | ||||
|         return results; | ||||
|     } | ||||
| 
 | ||||
|     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); | ||||
|         final List<Result> results = new ArrayList<>(); | ||||
|         final List<Thread> threads = new ArrayList<>(); | ||||
|         for (SRV record : result.getAnswersOrEmptySet()) { | ||||
|             if (record.name.length() == 0 && record.priority == 0) { | ||||
|                 continue; | ||||
|             } | ||||
|             threads.add(new Thread(() -> { | ||||
|                 final List<Result> ipv4s = resolveIp(record, A.class, result.isAuthenticData(), directTls); | ||||
|                 if (ipv4s.size() == 0) { | ||||
|             results.add(Result.fromRecord(record, directTls)); | ||||
|         } | ||||
| 
 | ||||
|         if (results.size() == 0) { | ||||
|             for (SRV record : result.getAnswersOrEmptySet()) { | ||||
|                 if (record.name.length() == 0 && record.priority == 0) { | ||||
|                     continue; | ||||
|                 } | ||||
|                 Result resolverResult = Result.fromRecord(record, directTls); | ||||
|                 resolverResult.authenticated = resolverResult.isAuthenticated(); | ||||
|                     ipv4s.add(resolverResult); | ||||
|                 results.add(resolverResult); | ||||
|             } | ||||
|                 synchronized (results) { | ||||
|                     results.addAll(ipv4s); | ||||
|         } | ||||
| 
 | ||||
|             })); | ||||
|             threads.add(new Thread(() -> { | ||||
|                 final List<Result> ipv6s = resolveIp(record, AAAA.class, result.isAuthenticData(), directTls); | ||||
|                 synchronized (results) { | ||||
|                     results.addAll(ipv6s); | ||||
|                 } | ||||
|             })); | ||||
|         } | ||||
|         for (Thread thread : threads) { | ||||
|             thread.start(); | ||||
|         } | ||||
|         for (Thread thread : threads) { | ||||
|             try { | ||||
|                 thread.join(); | ||||
|             } catch (InterruptedException e) { | ||||
|                 return Collections.emptyList(); | ||||
|             } | ||||
|         } | ||||
|         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) { | ||||
|         List<Result> results = new ArrayList<>(); | ||||
|         try { | ||||
|             for (A a : resolveWithFallback(dnsName, A.class, false).getAnswersOrEmptySet()) { | ||||
|                 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) { | ||||
|             if (withCnames) { | ||||
|                 for (CNAME cname : resolveWithFallback(dnsName, CNAME.class, false).getAnswersOrEmptySet()) { | ||||
|                     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); | ||||
|         } | ||||
|         results.add(Result.createDefault(dnsName)); | ||||
| 
 | ||||
|         Result result = new Result(); | ||||
|         result.hostname = dnsName; | ||||
|         result.port = 5222; | ||||
|         results.add(result); | ||||
| 
 | ||||
|         return results; | ||||
|     } | ||||
| 
 | ||||
|  | @ -262,18 +229,17 @@ public class Resolver { | |||
| 
 | ||||
|     public static class Result implements Comparable<Result> { | ||||
|         public static final String DOMAIN = "domain"; | ||||
|         public static final String IP = "ip"; | ||||
|         public static final String HOSTNAME = "hostname"; | ||||
|         public static final String PORT = "port"; | ||||
|         public static final String PRIORITY = "priority"; | ||||
|         public static final String DIRECT_TLS = "directTls"; | ||||
|         public static final String AUTHENTICATED = "authenticated"; | ||||
|         private InetAddress ip; | ||||
|         private DNSName hostname; | ||||
|         private int port = DEFAULT_PORT_XMPP; | ||||
|         private boolean directTls = false; | ||||
|         private boolean directTls = true; | ||||
|         private boolean authenticated = false; | ||||
|         private int priority; | ||||
|         private long rtt = -1; | ||||
| 
 | ||||
|         static Result fromRecord(SRV srv, boolean directTls) { | ||||
|             Result result = new Result(); | ||||
|  | @ -284,25 +250,15 @@ public class Resolver { | |||
|             return result; | ||||
|         } | ||||
| 
 | ||||
|         static Result createDefault(DNSName hostname, InetAddress ip) { | ||||
|         static Result createDefault(DNSName hostname) { | ||||
|             Result result = new Result(); | ||||
|             result.port = DEFAULT_PORT_XMPP; | ||||
|             result.hostname = hostname; | ||||
|             result.ip = ip; | ||||
|             return result; | ||||
|         } | ||||
| 
 | ||||
|         static Result createDefault(DNSName hostname) { | ||||
|             return createDefault(hostname, null); | ||||
|         } | ||||
| 
 | ||||
|         public static Result fromCursor(Cursor cursor) { | ||||
|             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)); | ||||
|             result.hostname = hostname == null ? null : DNSName.from(hostname); | ||||
|             result.port = cursor.getInt(cursor.getColumnIndex(PORT)); | ||||
|  | @ -323,14 +279,12 @@ public class Resolver { | |||
|             if (directTls != result.directTls) return false; | ||||
|             if (authenticated != result.authenticated) 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; | ||||
|         } | ||||
| 
 | ||||
|         @Override | ||||
|         public int hashCode() { | ||||
|             int result = ip != null ? ip.hashCode() : 0; | ||||
|             result = 31 * result + (hostname != null ? hostname.hashCode() : 0); | ||||
|             int result = hostname != null ? hostname.hashCode() : 0; | ||||
|             result = 31 * result + port; | ||||
|             result = 31 * result + (directTls ? 1 : 0); | ||||
|             result = 31 * result + (authenticated ? 1 : 0); | ||||
|  | @ -338,10 +292,6 @@ public class Resolver { | |||
|             return result; | ||||
|         } | ||||
| 
 | ||||
|         public InetAddress getIp() { | ||||
|             return ip; | ||||
|         } | ||||
| 
 | ||||
|         public int getPort() { | ||||
|             return port; | ||||
|         } | ||||
|  | @ -361,7 +311,6 @@ public class Resolver { | |||
|         @Override | ||||
|         public String toString() { | ||||
|             return "Result{" + | ||||
|                     "ip='" + (ip == null ? null : ip.getHostAddress()) + '\'' + | ||||
|                     ", hostame='" + hostname.toString() + '\'' + | ||||
|                     ", port=" + port + | ||||
|                     ", directTls=" + directTls + | ||||
|  | @ -374,17 +323,7 @@ public class Resolver { | |||
|         public int compareTo(@NonNull Result result) { | ||||
|             if (result.priority == priority) { | ||||
|                 if (directTls == result.directTls) { | ||||
|                     if (ip == null && result.ip == null) { | ||||
|                         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 { | ||||
|                     return directTls ? -1 : 1; | ||||
|                 } | ||||
|  | @ -395,7 +334,6 @@ public class Resolver { | |||
| 
 | ||||
|         public ContentValues toContentValues() { | ||||
|             final ContentValues contentValues = new ContentValues(); | ||||
|             contentValues.put(IP, ip == null ? null : ip.getAddress()); | ||||
|             contentValues.put(HOSTNAME, hostname == null ? null : hostname.toString()); | ||||
|             contentValues.put(PORT, port); | ||||
|             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.XmppConnectionService; | ||||
| import eu.siacs.conversations.utils.CryptoHelper; | ||||
| import eu.siacs.conversations.utils.IP; | ||||
| import eu.siacs.conversations.utils.Patterns; | ||||
| import eu.siacs.conversations.utils.Resolver; | ||||
| import eu.siacs.conversations.utils.SSLSocketHelper; | ||||
|  | @ -315,18 +314,10 @@ public class XmppConnection implements Runnable { | |||
|                         features.encryptionEnabled = result.isDirectTls(); | ||||
|                         verifiedHostname = result.isAuthenticated() ? result.getHostname().toString() : null; | ||||
|                         Log.d(Config.LOGTAG,"verified hostname "+verifiedHostname); | ||||
|                         final InetSocketAddress addr; | ||||
|                         if (result.getIp() != null) { | ||||
|                             addr = new InetSocketAddress(result.getIp(), result.getPort()); | ||||
|                             Log.d(Config.LOGTAG, account.getJid().asBareJid().toString() | ||||
|                                     + ": 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()); | ||||
|                         final InetSocketAddress 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) { | ||||
|                             localSocket = new Socket(); | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue