From 068c891703fc5a6da037776d05ee7712c505fb23 Mon Sep 17 00:00:00 2001 From: Kangping Dong Date: Tue, 13 Aug 2024 17:04:39 +0800 Subject: [PATCH] [android] handles unscoped ipv6 link-local address --- .../service/build.gradle | 2 +- .../service/BorderAgentDiscoverer.java | 55 ++++++++++++++++--- .../commissioner/service/BorderAgentInfo.java | 4 +- 3 files changed, 50 insertions(+), 11 deletions(-) diff --git a/android/openthread_commissioner/service/build.gradle b/android/openthread_commissioner/service/build.gradle index a210787c..ab1df350 100644 --- a/android/openthread_commissioner/service/build.gradle +++ b/android/openthread_commissioner/service/build.gradle @@ -73,7 +73,7 @@ dependencies { implementation fileTree(dir: "libs", include: ["*.jar"]) - implementation 'com.google.guava:guava:31.1-jre' + implementation 'com.google.guava:guava:33.2.1-android' implementation 'androidx.appcompat:appcompat:1.2.0' implementation "androidx.concurrent:concurrent-futures:1.1.0" implementation 'androidx.constraintlayout:constraintlayout:2.0.2' diff --git a/android/openthread_commissioner/service/src/main/java/io/openthread/commissioner/service/BorderAgentDiscoverer.java b/android/openthread_commissioner/service/src/main/java/io/openthread/commissioner/service/BorderAgentDiscoverer.java index 6e4214d1..47934d44 100644 --- a/android/openthread_commissioner/service/src/main/java/io/openthread/commissioner/service/BorderAgentDiscoverer.java +++ b/android/openthread_commissioner/service/src/main/java/io/openthread/commissioner/service/BorderAgentDiscoverer.java @@ -30,12 +30,17 @@ import android.Manifest.permission; import android.content.Context; +import android.net.ConnectivityManager; +import android.net.Network; import android.net.nsd.NsdManager; import android.net.nsd.NsdServiceInfo; import android.net.wifi.WifiManager; import android.util.Log; import androidx.annotation.Nullable; import androidx.annotation.RequiresPermission; +import com.google.common.net.InetAddresses; +import java.net.Inet6Address; +import java.net.InetAddress; import java.util.Map; import java.util.concurrent.ArrayBlockingQueue; import java.util.concurrent.BlockingQueue; @@ -52,13 +57,14 @@ public class BorderAgentDiscoverer implements NsdManager.DiscoveryListener { private static final String KEY_NETWORK_NAME = "nn"; private static final String KEY_EXTENDED_PAN_ID = "xp"; - private WifiManager.MulticastLock wifiMulticastLock; - private NsdManager nsdManager; - private BorderAgentListener borderAgentListener; + private final WifiManager.MulticastLock wifiMulticastLock; + private final NsdManager nsdManager; + private final ConnectivityManager connManager; + private final BorderAgentListener borderAgentListener; private ExecutorService executor = Executors.newSingleThreadExecutor(); - private BlockingQueue unresolvedServices = new ArrayBlockingQueue<>(256); - private AtomicBoolean isResolvingService = new AtomicBoolean(false); + private final BlockingQueue unresolvedServices = new ArrayBlockingQueue<>(256); + private final AtomicBoolean isResolvingService = new AtomicBoolean(false); private boolean isScanning = false; @@ -74,7 +80,8 @@ public BorderAgentDiscoverer(Context context, BorderAgentListener borderAgentLis WifiManager wifi = (WifiManager) context.getSystemService(Context.WIFI_SERVICE); wifiMulticastLock = wifi.createMulticastLock("multicastLock"); - nsdManager = (NsdManager) context.getSystemService(Context.NSD_SERVICE); + nsdManager = context.getSystemService(NsdManager.class); + connManager = context.getSystemService(ConnectivityManager.class); this.borderAgentListener = borderAgentListener; } @@ -112,7 +119,7 @@ public void onResolveFailed(NsdServiceInfo serviceInfo, int errorCode) { public void onServiceResolved(NsdServiceInfo serviceInfo) { BorderAgentInfo borderAgent = getBorderAgentInfo(serviceInfo); if (borderAgent != null) { - Log.d(TAG, "successfully resolved service: " + serviceInfo.toString()); + Log.d(TAG, "successfully resolved service: " + serviceInfo); Log.d( TAG, "successfully resolved service: " + serviceInfo.getHost().getCanonicalHostName()); @@ -214,10 +221,42 @@ private BorderAgentInfo getBorderAgentInfo(NsdServiceInfo serviceInfo) { id, new String(attrs.get(KEY_NETWORK_NAME)), attrs.get(KEY_EXTENDED_PAN_ID), - serviceInfo.getHost(), + handleNsdServiceAddress(serviceInfo.getHost()), serviceInfo.getPort()); } + /** + * Properly handles the {@link InetAddress} within a discovered {@link NsdServiceInfo}. + * + *

For example, adds the scope ID to an IPv6 link-local address to make is usable. + */ + private InetAddress handleNsdServiceAddress(InetAddress address) { + if (!(address instanceof Inet6Address)) { + return address; + } + + Inet6Address address6 = (Inet6Address) address; + + if (!address6.isLinkLocalAddress() || address6.getScopeId() != 0) { + return address6; + } + + // Sets the scope ID for IPv6 link-local address if it's missing. This can + // happen before Android U. + + // Assume the mDNS service is discovered on the current active default network + Network network = connManager.getActiveNetwork(); + if (network == null) { + return address6; + } + String interfaceName = connManager.getLinkProperties(network).getInterfaceName(); + if (interfaceName == null) { + return address6; + } + + return InetAddresses.forString(address6.getHostAddress() + "%" + interfaceName); + } + @Nullable private byte[] getBorderAgentId(NsdServiceInfo serviceInfo) { Map attrs = serviceInfo.getAttributes(); diff --git a/android/openthread_commissioner/service/src/main/java/io/openthread/commissioner/service/BorderAgentInfo.java b/android/openthread_commissioner/service/src/main/java/io/openthread/commissioner/service/BorderAgentInfo.java index 0a9babed..091a003f 100644 --- a/android/openthread_commissioner/service/src/main/java/io/openthread/commissioner/service/BorderAgentInfo.java +++ b/android/openthread_commissioner/service/src/main/java/io/openthread/commissioner/service/BorderAgentInfo.java @@ -48,9 +48,9 @@ public BorderAgentInfo( @NonNull byte[] extendedPanId, @NonNull InetAddress host, @NonNull int port) { - this.id = id.clone(); + this.id = (id == null ? null : id.clone()); this.networkName = networkName; - this.extendedPanId = extendedPanId.clone(); + this.extendedPanId = (extendedPanId == null ? null : extendedPanId.clone()); this.host = host; this.port = port; }