Merge "Create and implement a CacheListener inside WifiNetworkScoreCache." am: 31a8f712ab

am: 6d749fb489

Change-Id: I711dd2b9d142a3a92ed641b08dd944be329e503a
This commit is contained in:
Sundeep Ghuman
2016-12-21 00:51:49 +00:00
committed by android-build-merger
2 changed files with 232 additions and 104 deletions

View File

@@ -17,13 +17,19 @@
package android.net.wifi;
import android.Manifest.permission;
import android.annotation.NonNull;
import android.annotation.Nullable;
import android.annotation.SystemApi;
import android.content.Context;
import android.os.Handler;
import android.net.INetworkScoreCache;
import android.net.NetworkKey;
import android.net.ScoredNetwork;
import android.util.Log;
import com.android.internal.util.Preconditions;
import com.android.internal.annotations.GuardedBy;
import java.io.FileDescriptor;
import java.io.PrintWriter;
import java.util.HashMap;
@@ -43,30 +49,55 @@ public class WifiNetworkScoreCache extends INetworkScoreCache.Stub {
// We treat the lowest possible score as though there were no score, effectively allowing the
// scorer to provide an RSSI threshold below which a network should not be used.
public static final int INVALID_NETWORK_SCORE = Byte.MIN_VALUE;
// See {@link #CacheListener}.
@Nullable
@GuardedBy("mCacheLock")
private CacheListener mListener;
private final Context mContext;
private final Object mCacheLock = new Object();
// The key is of the form "<ssid>"<bssid>
// TODO: What about SSIDs that can't be encoded as UTF-8?
private final Map<String, ScoredNetwork> mNetworkCache;
public WifiNetworkScoreCache(Context context) {
mContext = context;
this(context, null /* listener */);
}
/**
* Instantiates a WifiNetworkScoreCache.
*
* @param context Application context
* @param listener CacheListener for cache updates
*/
public WifiNetworkScoreCache(Context context, @Nullable CacheListener listener) {
mContext = context.getApplicationContext();
mListener = listener;
mNetworkCache = new HashMap<String, ScoredNetwork>();
}
@Override public final void updateScores(List<ScoredNetwork> networks) {
if (networks == null) {
if (networks == null || networks.isEmpty()) {
return;
}
Log.e(TAG, "updateScores list size=" + networks.size());
}
Log.d(TAG, "updateScores list size=" + networks.size());
synchronized(mNetworkCache) {
for (ScoredNetwork network : networks) {
String networkKey = buildNetworkKey(network);
if (networkKey == null) continue;
mNetworkCache.put(networkKey, network);
}
}
synchronized(mNetworkCache) {
for (ScoredNetwork network : networks) {
String networkKey = buildNetworkKey(network);
if (networkKey == null) continue;
mNetworkCache.put(networkKey, network);
}
}
synchronized (mCacheLock) {
if (mListener != null) {
mListener.post(networks);
}
}
}
@Override public final void clearScores() {
@@ -193,4 +224,53 @@ public class WifiNetworkScoreCache extends INetworkScoreCache.Stub {
}
}
/** Registers a CacheListener instance, replacing the previous listener if it existed. */
public void registerListener(CacheListener listener) {
synchronized (mCacheLock) {
mListener = listener;
}
}
/** Removes the registered CacheListener. */
public void unregisterListener() {
synchronized (mCacheLock) {
mListener = null;
}
}
/** Listener for updates to the cache inside WifiNetworkScoreCache. */
public abstract static class CacheListener {
private Handler mHandler;
/**
* Constructor for CacheListener.
*
* @param handler the Handler on which to invoke the {@link #networkCacheUpdated} method.
* This cannot be null.
*/
public CacheListener(@NonNull Handler handler) {
Preconditions.checkNotNull(handler);
mHandler = handler;
}
/** Invokes the {@link #networkCacheUpdated(List<ScoredNetwork>)} method on the handler. */
void post(List<ScoredNetwork> updatedNetworks) {
mHandler.post(new Runnable() {
@Override
public void run() {
networkCacheUpdated(updatedNetworks);
}
});
}
/**
* Invoked whenever the cache is updated.
*
* <p>Clearing the cache does not invoke this method.
*
* @param updatedNetworks the networks that were updated
*/
public abstract void networkCacheUpdated(List<ScoredNetwork> updatedNetworks);
}
}

View File

@@ -17,6 +17,8 @@
package android.net.wifi;
import static org.junit.Assert.*;
import static org.mockito.Mockito.timeout;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
import android.content.Context;
@@ -24,6 +26,9 @@ import android.net.NetworkKey;
import android.net.RssiCurve;
import android.net.ScoredNetwork;
import android.net.WifiKey;
import android.net.wifi.WifiNetworkScoreCache.CacheListener;
import android.os.Handler;
import android.os.HandlerThread;
import android.support.test.filters.SmallTest;
import android.support.test.runner.AndroidJUnit4;
@@ -33,124 +38,167 @@ import org.junit.Rule;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.ArgumentCaptor;
import org.mockito.Captor;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
/** Unit tests for {@link WifiNetworkScoreCache}. */
@RunWith(AndroidJUnit4.class)
@SmallTest
public class WifiNetworkScoreCacheTest {
@Mock public Context mockContext; // isn't used, can be null
@Mock private RssiCurve mockRssiCurve;
public static final String SSID = "ssid";
public static final String FORMATTED_SSID = "\"" + SSID + "\"";
public static final String BSSID = "AA:AA:AA:AA:AA:AA";
public static final String SSID = "ssid";
public static final String FORMATTED_SSID = "\"" + SSID + "\"";
public static final String BSSID = "AA:AA:AA:AA:AA:AA";
public static final WifiKey VALID_KEY = new WifiKey(FORMATTED_SSID, BSSID);
public static final WifiKey VALID_KEY = new WifiKey(FORMATTED_SSID, BSSID);
public static final ScanResult VALID_SCAN_RESULT = buildScanResult(SSID, BSSID);
public static final ScanResult VALID_SCAN_RESULT = buildScanResult(SSID, BSSID);
private ScoredNetwork mValidScoredNetwork;
private WifiNetworkScoreCache mScoreCache =
new WifiNetworkScoreCache(mockContext);
private static ScanResult buildScanResult(String ssid, String bssid) {
return new ScanResult(
WifiSsid.createFromAsciiEncoded(ssid),
bssid,
"" /* caps */,
0 /* level */,
0 /* frequency */,
0 /* tsf */,
0 /* distCm */,
0 /* distSdCm*/);
}
private static ScoredNetwork buildScoredNetwork(WifiKey key, RssiCurve curve) {
return new ScoredNetwork(new NetworkKey(key), curve);
}
// Called from setup
private void initializeCacheWithValidScoredNetwork() {
mScoreCache.updateScores(ImmutableList.of(mValidScoredNetwork));
}
@Before
public void setUp() {
MockitoAnnotations.initMocks(this);
mValidScoredNetwork = buildScoredNetwork(VALID_KEY, mockRssiCurve);
mScoreCache = new WifiNetworkScoreCache(mockContext);
initializeCacheWithValidScoredNetwork();
}
@Mock private Context mockApplicationContext;
@Mock private Context mockContext; // isn't used, can be null
@Mock private RssiCurve mockRssiCurve;
@Test
public void isScoredNetworkShouldReturnTrueAfterUpdateScoresIsCalled() {
assertTrue(mScoreCache.isScoredNetwork(VALID_SCAN_RESULT));
}
private CacheListener mCacheListener;
private CountDownLatch mLatch;
private Handler mHandler;
private List<ScoredNetwork> mUpdatedNetworksCaptor;
private ScoredNetwork mValidScoredNetwork;
private WifiNetworkScoreCache mScoreCache =
new WifiNetworkScoreCache(mockContext);
@Test
public void isScoredNetworkShouldReturnFalseAfterClearScoresIsCalled() {
mScoreCache.clearScores();
assertFalse(mScoreCache.isScoredNetwork(VALID_SCAN_RESULT));
}
private static ScanResult buildScanResult(String ssid, String bssid) {
return new ScanResult(
WifiSsid.createFromAsciiEncoded(ssid),
bssid,
"" /* caps */,
0 /* level */,
0 /* frequency */,
0 /* tsf */,
0 /* distCm */,
0 /* distSdCm*/);
}
@Test
public void updateScoresShouldAddNewNetwork() {
WifiKey key2 = new WifiKey("\"ssid2\"", BSSID);
ScoredNetwork network2 = buildScoredNetwork(key2, mockRssiCurve);
ScanResult result2 = buildScanResult("ssid2", BSSID);
private static ScoredNetwork buildScoredNetwork(WifiKey key, RssiCurve curve) {
return new ScoredNetwork(new NetworkKey(key), curve);
}
mScoreCache.updateScores(ImmutableList.of(network2));
// Called from setup
private void initializeCacheWithValidScoredNetwork() {
mScoreCache.updateScores(ImmutableList.of(mValidScoredNetwork));
}
assertTrue(mScoreCache.isScoredNetwork(VALID_SCAN_RESULT));
assertTrue(mScoreCache.isScoredNetwork(result2));
}
@Before
public void setUp() {
MockitoAnnotations.initMocks(this);
@Test
public void hasScoreCurveShouldReturnTrue() {
assertTrue(mScoreCache.hasScoreCurve(VALID_SCAN_RESULT));
}
when(mockContext.getApplicationContext()).thenReturn(mockApplicationContext);
@Test
public void hasScoreCurveShouldReturnFalseWhenNoCachedNetwork() {
ScanResult unscored = buildScanResult("fake", BSSID);
assertFalse(mScoreCache.hasScoreCurve(unscored));
}
mValidScoredNetwork = buildScoredNetwork(VALID_KEY, mockRssiCurve);
mScoreCache = new WifiNetworkScoreCache(mockContext);
initializeCacheWithValidScoredNetwork();
@Test
public void hasScoreCurveShouldReturnFalseWhenScoredNetworkHasNoCurve() {
ScoredNetwork noCurve = buildScoredNetwork(VALID_KEY, null /* rssiCurve */);
mScoreCache.updateScores(ImmutableList.of(noCurve));
HandlerThread thread = new HandlerThread("WifiNetworkScoreCacheTest Handler Thread");
thread.start();
mHandler = new Handler(thread.getLooper());
mLatch = new CountDownLatch(1);
mCacheListener = new CacheListener(mHandler) {
@Override
public void networkCacheUpdated(List<ScoredNetwork> updatedNetworks) {
mUpdatedNetworksCaptor = updatedNetworks;
mLatch.countDown();
}
};
}
assertFalse(mScoreCache.hasScoreCurve(VALID_SCAN_RESULT));
}
@Test
public void getNetworkScoreShouldReturnScore() {
final byte score = 50;
final int rssi = -70;
ScanResult result = new ScanResult(VALID_SCAN_RESULT);
result.level = rssi;
@Test
public void isScoredNetworkShouldReturnTrueAfterUpdateScoresIsCalled() {
assertTrue(mScoreCache.isScoredNetwork(VALID_SCAN_RESULT));
}
when(mockRssiCurve.lookupScore(rssi)).thenReturn(score);
@Test
public void isScoredNetworkShouldReturnFalseAfterClearScoresIsCalled() {
mScoreCache.clearScores();
assertFalse(mScoreCache.isScoredNetwork(VALID_SCAN_RESULT));
}
assertEquals(score, mScoreCache.getNetworkScore(result));
}
@Test
public void updateScoresShouldAddNewNetwork() {
WifiKey key2 = new WifiKey("\"ssid2\"", BSSID);
ScoredNetwork network2 = buildScoredNetwork(key2, mockRssiCurve);
ScanResult result2 = buildScanResult("ssid2", BSSID);
@Test
public void getMeteredHintShouldReturnFalse() {
assertFalse(mScoreCache.getMeteredHint(VALID_SCAN_RESULT));
}
mScoreCache.updateScores(ImmutableList.of(network2));
@Test
public void getMeteredHintShouldReturnTrue() {
ScoredNetwork network =
new ScoredNetwork(new NetworkKey(VALID_KEY), mockRssiCurve, true /* metered Hint */);
mScoreCache.updateScores(ImmutableList.of(network));
assertTrue(mScoreCache.isScoredNetwork(VALID_SCAN_RESULT));
assertTrue(mScoreCache.isScoredNetwork(result2));
}
assertTrue(mScoreCache.getMeteredHint(VALID_SCAN_RESULT));
}
@Test
public void hasScoreCurveShouldReturnTrue() {
assertTrue(mScoreCache.hasScoreCurve(VALID_SCAN_RESULT));
}
@Test
public void hasScoreCurveShouldReturnFalseWhenNoCachedNetwork() {
ScanResult unscored = buildScanResult("fake", BSSID);
assertFalse(mScoreCache.hasScoreCurve(unscored));
}
@Test
public void hasScoreCurveShouldReturnFalseWhenScoredNetworkHasNoCurve() {
ScoredNetwork noCurve = buildScoredNetwork(VALID_KEY, null /* rssiCurve */);
mScoreCache.updateScores(ImmutableList.of(noCurve));
assertFalse(mScoreCache.hasScoreCurve(VALID_SCAN_RESULT));
}
@Test
public void getNetworkScoreShouldReturnScore() {
final byte score = 50;
final int rssi = -70;
ScanResult result = new ScanResult(VALID_SCAN_RESULT);
result.level = rssi;
when(mockRssiCurve.lookupScore(rssi)).thenReturn(score);
assertEquals(score, mScoreCache.getNetworkScore(result));
}
@Test
public void getMeteredHintShouldReturnFalse() {
assertFalse(mScoreCache.getMeteredHint(VALID_SCAN_RESULT));
}
@Test
public void getMeteredHintShouldReturnTrue() {
ScoredNetwork network =
new ScoredNetwork(
new NetworkKey(VALID_KEY), mockRssiCurve, true /* metered Hint */);
mScoreCache.updateScores(ImmutableList.of(network));
assertTrue(mScoreCache.getMeteredHint(VALID_SCAN_RESULT));
}
@Test
public void updateScoresShouldInvokeCacheListener_networkCacheUpdated() {
mScoreCache = new WifiNetworkScoreCache(mockContext, mCacheListener);
initializeCacheWithValidScoredNetwork();
try {
mLatch.await(1, TimeUnit.SECONDS); // wait for listener to be executed
} catch (InterruptedException e) {
fail("Interrupted Exception while waiting for listener to be invoked.");
}
assertEquals("One network should be updated", 1, mUpdatedNetworksCaptor.size());
assertEquals(mValidScoredNetwork, mUpdatedNetworksCaptor.get(0));
}
}