Genericize NC#hasSameUids

This will be used by another set of UIDs in a future patch

Test: FrameworksNetTests
Change-Id: I2c5d18ef93e73b702723814592ef3f3baf5dfbc4
This commit is contained in:
Chalard Jean
2021-12-13 21:37:12 +09:00
parent 286c0e5336
commit f4802fa4c2
4 changed files with 88 additions and 25 deletions

View File

@@ -1591,28 +1591,6 @@ public final class NetworkCapabilities implements Parcelable {
return false; return false;
} }
/**
* Compare if the given NetworkCapabilities have the same UIDs.
*
* @hide
*/
public static boolean hasSameUids(@Nullable NetworkCapabilities nc1,
@Nullable NetworkCapabilities nc2) {
final Set<UidRange> uids1 = (nc1 == null) ? null : nc1.mUids;
final Set<UidRange> uids2 = (nc2 == null) ? null : nc2.mUids;
if (null == uids1) return null == uids2;
if (null == uids2) return false;
// Make a copy so it can be mutated to check that all ranges in uids2 also are in uids.
final Set<UidRange> uids = new ArraySet<>(uids2);
for (UidRange range : uids1) {
if (!uids.contains(range)) {
return false;
}
uids.remove(range);
}
return uids.isEmpty();
}
/** /**
* Tests if the set of UIDs that this network applies to is the same as the passed network. * Tests if the set of UIDs that this network applies to is the same as the passed network.
* <p> * <p>
@@ -1623,13 +1601,13 @@ public final class NetworkCapabilities implements Parcelable {
* Note that this method is not very optimized, which is fine as long as it's not used very * Note that this method is not very optimized, which is fine as long as it's not used very
* often. * often.
* <p> * <p>
* nc is assumed nonnull. * nc is assumed nonnull, else NPE.
* *
* @hide * @hide
*/ */
@VisibleForTesting @VisibleForTesting
public boolean equalsUids(@NonNull NetworkCapabilities nc) { public boolean equalsUids(@NonNull NetworkCapabilities nc) {
return hasSameUids(nc, this); return UidRange.hasSameUids(nc.mUids, mUids);
} }
/** /**

View File

@@ -180,4 +180,24 @@ public final class UidRange implements Parcelable {
} }
return uids; return uids;
} }
/**
* Compare if the given UID range sets have the same UIDs.
*
* @hide
*/
public static boolean hasSameUids(@Nullable Set<UidRange> uids1,
@Nullable Set<UidRange> uids2) {
if (null == uids1) return null == uids2;
if (null == uids2) return false;
// Make a copy so it can be mutated to check that all ranges in uids2 also are in uids.
final Set<UidRange> remainingUids = new ArraySet<>(uids2);
for (UidRange range : uids1) {
if (!remainingUids.contains(range)) {
return false;
}
remainingUids.remove(range);
}
return remainingUids.isEmpty();
}
} }

View File

@@ -7705,7 +7705,9 @@ public class ConnectivityService extends IConnectivityManager.Stub
// changed. // changed.
// TODO: Try to track the default network that apps use and only send a proxy broadcast when // TODO: Try to track the default network that apps use and only send a proxy broadcast when
// that happens to prevent false alarms. // that happens to prevent false alarms.
if (nai.isVPN() && nai.everConnected && !NetworkCapabilities.hasSameUids(prevNc, newNc) final Set<UidRange> prevUids = prevNc == null ? null : prevNc.getUidRanges();
final Set<UidRange> newUids = newNc == null ? null : newNc.getUidRanges();
if (nai.isVPN() && nai.everConnected && !UidRange.hasSameUids(prevUids, newUids)
&& (nai.linkProperties.getHttpProxy() != null || isProxySetOnAnyDefaultNetwork())) { && (nai.linkProperties.getHttpProxy() != null || isProxySetOnAnyDefaultNetwork())) {
mProxyTracker.sendProxyBroadcast(); mProxyTracker.sendProxyBroadcast();
} }

View File

@@ -22,11 +22,15 @@ import static android.os.UserHandle.USER_SYSTEM;
import static android.os.UserHandle.getUid; import static android.os.UserHandle.getUid;
import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertTrue; import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail; import static org.junit.Assert.fail;
import android.annotation.NonNull;
import android.annotation.Nullable;
import android.os.Build; import android.os.Build;
import android.os.UserHandle; import android.os.UserHandle;
import android.util.ArraySet;
import androidx.test.filters.SmallTest; import androidx.test.filters.SmallTest;
import androidx.test.runner.AndroidJUnit4; import androidx.test.runner.AndroidJUnit4;
@@ -38,6 +42,8 @@ import org.junit.Rule;
import org.junit.Test; import org.junit.Test;
import org.junit.runner.RunWith; import org.junit.runner.RunWith;
import java.util.Set;
@RunWith(AndroidJUnit4.class) @RunWith(AndroidJUnit4.class)
@SmallTest @SmallTest
public class UidRangeTest { public class UidRangeTest {
@@ -110,4 +116,61 @@ public class UidRangeTest {
assertEquals(USER_SYSTEM + 1, uidRangeOfSecondaryUser.getStartUser()); assertEquals(USER_SYSTEM + 1, uidRangeOfSecondaryUser.getStartUser());
assertEquals(USER_SYSTEM + 1, uidRangeOfSecondaryUser.getEndUser()); assertEquals(USER_SYSTEM + 1, uidRangeOfSecondaryUser.getEndUser());
} }
private static void assertSameUids(@NonNull final String msg, @Nullable final Set<UidRange> s1,
@Nullable final Set<UidRange> s2) {
assertTrue(msg + " : " + s1 + " unexpectedly different from " + s2,
UidRange.hasSameUids(s1, s2));
}
private static void assertDifferentUids(@NonNull final String msg,
@Nullable final Set<UidRange> s1, @Nullable final Set<UidRange> s2) {
assertFalse(msg + " : " + s1 + " unexpectedly equal to " + s2,
UidRange.hasSameUids(s1, s2));
}
// R doesn't have UidRange.hasSameUids, but since S has the module, it does have hasSameUids.
@Test @IgnoreUpTo(Build.VERSION_CODES.R)
public void testHasSameUids() {
final UidRange uids1 = new UidRange(1, 100);
final UidRange uids2 = new UidRange(3, 300);
final UidRange uids3 = new UidRange(1, 1000);
final UidRange uids4 = new UidRange(800, 1000);
assertSameUids("null <=> null", null, null);
final Set<UidRange> set1 = new ArraySet<>();
assertDifferentUids("empty <=> null", set1, null);
final Set<UidRange> set2 = new ArraySet<>();
set1.add(uids1);
assertDifferentUids("uids1 <=> null", set1, null);
assertDifferentUids("null <=> uids1", null, set1);
assertDifferentUids("uids1 <=> empty", set1, set2);
set2.add(uids1);
assertSameUids("uids1 <=> uids1", set1, set2);
set1.add(uids2);
assertDifferentUids("uids1,2 <=> uids1", set1, set2);
set1.add(uids3);
assertDifferentUids("uids1,2,3 <=> uids1", set1, set2);
set2.add(uids3);
assertDifferentUids("uids1,2,3 <=> uids1,3", set1, set2);
set2.add(uids2);
assertSameUids("uids1,2,3 <=> uids1,2,3", set1, set2);
set1.remove(uids2);
assertDifferentUids("uids1,3 <=> uids1,2,3", set1, set2);
set1.add(uids4);
assertDifferentUids("uids1,3,4 <=> uids1,2,3", set1, set2);
set2.add(uids4);
assertDifferentUids("uids1,3,4 <=> uids1,2,3,4", set1, set2);
assertDifferentUids("uids1,3,4 <=> null", set1, null);
set2.remove(uids2);
assertSameUids("uids1,3,4 <=> uids1,3,4", set1, set2);
set2.remove(uids1);
assertDifferentUids("uids1,3,4 <=> uids3,4", set1, set2);
set2.remove(uids3);
assertDifferentUids("uids1,3,4 <=> uids4", set1, set2);
set2.remove(uids4);
assertDifferentUids("uids1,3,4 <=> empty", set1, set2);
assertDifferentUids("null <=> empty", null, set2);
assertSameUids("empty <=> empty", set2, new ArraySet<>());
}
} }