Network switch notifications: rate & daily limits am: 9bf6fef270 am: 858499b2c0 am: 2c535a4f7c
am: e6c813b201
Change-Id: I0c067ec355eb2a50af3c828b1fe3ee8a78e44536
This commit is contained in:
@@ -838,7 +838,14 @@ public class ConnectivityService extends IConnectivityManager.Stub
|
|||||||
mKeepaliveTracker = new KeepaliveTracker(mHandler);
|
mKeepaliveTracker = new KeepaliveTracker(mHandler);
|
||||||
mNotifier = new NetworkNotificationManager(mContext, mTelephonyManager,
|
mNotifier = new NetworkNotificationManager(mContext, mTelephonyManager,
|
||||||
mContext.getSystemService(NotificationManager.class));
|
mContext.getSystemService(NotificationManager.class));
|
||||||
mLingerMonitor = new LingerMonitor(mContext, mNotifier);
|
|
||||||
|
final int dailyLimit = Settings.Global.getInt(mContext.getContentResolver(),
|
||||||
|
Settings.Global.NETWORK_SWITCH_NOTIFICATION_DAILY_LIMIT,
|
||||||
|
LingerMonitor.DEFAULT_NOTIFICATION_DAILY_LIMIT);
|
||||||
|
final long rateLimit = Settings.Global.getLong(mContext.getContentResolver(),
|
||||||
|
Settings.Global.NETWORK_SWITCH_NOTIFICATION_RATE_LIMIT_MILLIS,
|
||||||
|
LingerMonitor.DEFAULT_NOTIFICATION_RATE_LIMIT_MILLIS);
|
||||||
|
mLingerMonitor = new LingerMonitor(mContext, mNotifier, dailyLimit, rateLimit);
|
||||||
}
|
}
|
||||||
|
|
||||||
private NetworkRequest createInternetRequestForTransport(int transportType) {
|
private NetworkRequest createInternetRequestForTransport(int transportType) {
|
||||||
|
|||||||
@@ -17,12 +17,14 @@
|
|||||||
package com.android.server.connectivity;
|
package com.android.server.connectivity;
|
||||||
|
|
||||||
import android.app.PendingIntent;
|
import android.app.PendingIntent;
|
||||||
import android.net.NetworkCapabilities;
|
|
||||||
import android.content.ComponentName;
|
import android.content.ComponentName;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
|
import android.net.NetworkCapabilities;
|
||||||
|
import android.os.SystemClock;
|
||||||
import android.os.UserHandle;
|
import android.os.UserHandle;
|
||||||
import android.text.TextUtils;
|
import android.text.TextUtils;
|
||||||
|
import android.text.format.DateUtils;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
import android.util.SparseArray;
|
import android.util.SparseArray;
|
||||||
import android.util.SparseIntArray;
|
import android.util.SparseIntArray;
|
||||||
@@ -30,6 +32,7 @@ import android.util.SparseBooleanArray;
|
|||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
|
||||||
|
import com.android.internal.R;
|
||||||
import com.android.internal.annotations.VisibleForTesting;
|
import com.android.internal.annotations.VisibleForTesting;
|
||||||
import com.android.internal.util.MessageUtils;
|
import com.android.internal.util.MessageUtils;
|
||||||
import com.android.server.connectivity.NetworkNotificationManager;
|
import com.android.server.connectivity.NetworkNotificationManager;
|
||||||
@@ -50,6 +53,9 @@ public class LingerMonitor {
|
|||||||
private static final boolean VDBG = false;
|
private static final boolean VDBG = false;
|
||||||
private static final String TAG = LingerMonitor.class.getSimpleName();
|
private static final String TAG = LingerMonitor.class.getSimpleName();
|
||||||
|
|
||||||
|
public static final int DEFAULT_NOTIFICATION_DAILY_LIMIT = 3;
|
||||||
|
public static final long DEFAULT_NOTIFICATION_RATE_LIMIT_MILLIS = DateUtils.MINUTE_IN_MILLIS;
|
||||||
|
|
||||||
private static final HashMap<String, Integer> TRANSPORT_NAMES = makeTransportToNameMap();
|
private static final HashMap<String, Integer> TRANSPORT_NAMES = makeTransportToNameMap();
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
public static final Intent CELLULAR_SETTINGS = new Intent().setComponent(new ComponentName(
|
public static final Intent CELLULAR_SETTINGS = new Intent().setComponent(new ComponentName(
|
||||||
@@ -65,6 +71,12 @@ public class LingerMonitor {
|
|||||||
|
|
||||||
private final Context mContext;
|
private final Context mContext;
|
||||||
private final NetworkNotificationManager mNotifier;
|
private final NetworkNotificationManager mNotifier;
|
||||||
|
private final int mDailyLimit;
|
||||||
|
private final long mRateLimitMillis;
|
||||||
|
|
||||||
|
private long mFirstNotificationMillis;
|
||||||
|
private long mLastNotificationMillis;
|
||||||
|
private int mNotificationCounter;
|
||||||
|
|
||||||
/** Current notifications. Maps the netId we switched away from to the netId we switched to. */
|
/** Current notifications. Maps the netId we switched away from to the netId we switched to. */
|
||||||
private final SparseIntArray mNotifications = new SparseIntArray();
|
private final SparseIntArray mNotifications = new SparseIntArray();
|
||||||
@@ -72,9 +84,12 @@ public class LingerMonitor {
|
|||||||
/** Whether we ever notified that we switched away from a particular network. */
|
/** Whether we ever notified that we switched away from a particular network. */
|
||||||
private final SparseBooleanArray mEverNotified = new SparseBooleanArray();
|
private final SparseBooleanArray mEverNotified = new SparseBooleanArray();
|
||||||
|
|
||||||
public LingerMonitor(Context context, NetworkNotificationManager notifier) {
|
public LingerMonitor(Context context, NetworkNotificationManager notifier,
|
||||||
|
int dailyLimit, long rateLimitMillis) {
|
||||||
mContext = context;
|
mContext = context;
|
||||||
mNotifier = notifier;
|
mNotifier = notifier;
|
||||||
|
mDailyLimit = dailyLimit;
|
||||||
|
mRateLimitMillis = rateLimitMillis;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static HashMap<String, Integer> makeTransportToNameMap() {
|
private static HashMap<String, Integer> makeTransportToNameMap() {
|
||||||
@@ -109,8 +124,8 @@ public class LingerMonitor {
|
|||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
public boolean isNotificationEnabled(NetworkAgentInfo fromNai, NetworkAgentInfo toNai) {
|
public boolean isNotificationEnabled(NetworkAgentInfo fromNai, NetworkAgentInfo toNai) {
|
||||||
// TODO: Evaluate moving to CarrierConfigManager.
|
// TODO: Evaluate moving to CarrierConfigManager.
|
||||||
String[] notifySwitches = mContext.getResources().getStringArray(
|
String[] notifySwitches =
|
||||||
com.android.internal.R.array.config_networkNotifySwitches);
|
mContext.getResources().getStringArray(R.array.config_networkNotifySwitches);
|
||||||
|
|
||||||
if (VDBG) {
|
if (VDBG) {
|
||||||
Log.d(TAG, "Notify on network switches: " + Arrays.toString(notifySwitches));
|
Log.d(TAG, "Notify on network switches: " + Arrays.toString(notifySwitches));
|
||||||
@@ -156,42 +171,38 @@ public class LingerMonitor {
|
|||||||
|
|
||||||
// Notify the user of a network switch using a notification or a toast.
|
// Notify the user of a network switch using a notification or a toast.
|
||||||
private void notify(NetworkAgentInfo fromNai, NetworkAgentInfo toNai, boolean forceToast) {
|
private void notify(NetworkAgentInfo fromNai, NetworkAgentInfo toNai, boolean forceToast) {
|
||||||
boolean notify = false;
|
int notifyType =
|
||||||
int notifyType = mContext.getResources().getInteger(
|
mContext.getResources().getInteger(R.integer.config_networkNotifySwitchType);
|
||||||
com.android.internal.R.integer.config_networkNotifySwitchType);
|
|
||||||
|
|
||||||
if (notifyType == NOTIFY_TYPE_NOTIFICATION && forceToast) {
|
if (notifyType == NOTIFY_TYPE_NOTIFICATION && forceToast) {
|
||||||
notifyType = NOTIFY_TYPE_TOAST;
|
notifyType = NOTIFY_TYPE_TOAST;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (notifyType) {
|
|
||||||
case NOTIFY_TYPE_NONE:
|
|
||||||
break;
|
|
||||||
case NOTIFY_TYPE_NOTIFICATION:
|
|
||||||
showNotification(fromNai, toNai);
|
|
||||||
notify = true;
|
|
||||||
break;
|
|
||||||
case NOTIFY_TYPE_TOAST:
|
|
||||||
mNotifier.showToast(fromNai, toNai);
|
|
||||||
notify = true;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
Log.e(TAG, "Unknown notify type " + notifyType);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (VDBG) {
|
if (VDBG) {
|
||||||
Log.d(TAG, "Notify type: " + sNotifyTypeNames.get(notifyType, "" + notifyType));
|
Log.d(TAG, "Notify type: " + sNotifyTypeNames.get(notifyType, "" + notifyType));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (notify) {
|
switch (notifyType) {
|
||||||
|
case NOTIFY_TYPE_NONE:
|
||||||
|
return;
|
||||||
|
case NOTIFY_TYPE_NOTIFICATION:
|
||||||
|
showNotification(fromNai, toNai);
|
||||||
|
break;
|
||||||
|
case NOTIFY_TYPE_TOAST:
|
||||||
|
mNotifier.showToast(fromNai, toNai);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
Log.e(TAG, "Unknown notify type " + notifyType);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (DBG) {
|
if (DBG) {
|
||||||
Log.d(TAG, "Notifying switch from=" + fromNai.name() + " to=" + toNai.name() +
|
Log.d(TAG, "Notifying switch from=" + fromNai.name() + " to=" + toNai.name() +
|
||||||
" type=" + sNotifyTypeNames.get(notifyType, "unknown(" + notifyType + ")"));
|
" type=" + sNotifyTypeNames.get(notifyType, "unknown(" + notifyType + ")"));
|
||||||
}
|
}
|
||||||
|
|
||||||
mNotifications.put(fromNai.network.netId, toNai.network.netId);
|
mNotifications.put(fromNai.network.netId, toNai.network.netId);
|
||||||
mEverNotified.put(fromNai.network.netId, true);
|
mEverNotified.put(fromNai.network.netId, true);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// The default network changed from fromNai to toNai due to a change in score.
|
// The default network changed from fromNai to toNai due to a change in score.
|
||||||
public void noteLingerDefaultNetwork(NetworkAgentInfo fromNai, NetworkAgentInfo toNai) {
|
public void noteLingerDefaultNetwork(NetworkAgentInfo fromNai, NetworkAgentInfo toNai) {
|
||||||
@@ -251,10 +262,13 @@ public class LingerMonitor {
|
|||||||
// unvalidated.
|
// unvalidated.
|
||||||
if (fromNai.lastValidated) return;
|
if (fromNai.lastValidated) return;
|
||||||
|
|
||||||
if (isNotificationEnabled(fromNai, toNai)) {
|
if (!isNotificationEnabled(fromNai, toNai)) return;
|
||||||
|
|
||||||
|
final long now = SystemClock.elapsedRealtime();
|
||||||
|
if (isRateLimited(now) || isAboveDailyLimit(now)) return;
|
||||||
|
|
||||||
notify(fromNai, toNai, forceToast);
|
notify(fromNai, toNai, forceToast);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
public void noteDisconnect(NetworkAgentInfo nai) {
|
public void noteDisconnect(NetworkAgentInfo nai) {
|
||||||
mNotifications.delete(nai.network.netId);
|
mNotifications.delete(nai.network.netId);
|
||||||
@@ -262,4 +276,29 @@ public class LingerMonitor {
|
|||||||
maybeStopNotifying(nai);
|
maybeStopNotifying(nai);
|
||||||
// No need to cancel notifications on nai: NetworkMonitor does that on disconnect.
|
// No need to cancel notifications on nai: NetworkMonitor does that on disconnect.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private boolean isRateLimited(long now) {
|
||||||
|
final long millisSinceLast = now - mLastNotificationMillis;
|
||||||
|
if (millisSinceLast < mRateLimitMillis) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
mLastNotificationMillis = now;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isAboveDailyLimit(long now) {
|
||||||
|
if (mFirstNotificationMillis == 0) {
|
||||||
|
mFirstNotificationMillis = now;
|
||||||
|
}
|
||||||
|
final long millisSinceFirst = now - mFirstNotificationMillis;
|
||||||
|
if (millisSinceFirst > DateUtils.DAY_IN_MILLIS) {
|
||||||
|
mNotificationCounter = 0;
|
||||||
|
mFirstNotificationMillis = 0;
|
||||||
|
}
|
||||||
|
if (mNotificationCounter >= mDailyLimit) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
mNotificationCounter++;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ import android.net.Network;
|
|||||||
import android.net.NetworkCapabilities;
|
import android.net.NetworkCapabilities;
|
||||||
import android.net.NetworkInfo;
|
import android.net.NetworkInfo;
|
||||||
import android.net.NetworkMisc;
|
import android.net.NetworkMisc;
|
||||||
|
import android.text.format.DateUtils;
|
||||||
import com.android.internal.R;
|
import com.android.internal.R;
|
||||||
import com.android.server.ConnectivityService;
|
import com.android.server.ConnectivityService;
|
||||||
import com.android.server.connectivity.NetworkNotificationManager;
|
import com.android.server.connectivity.NetworkNotificationManager;
|
||||||
@@ -40,11 +41,18 @@ import static org.mockito.Mockito.never;
|
|||||||
import static org.mockito.Mockito.times;
|
import static org.mockito.Mockito.times;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
import static org.mockito.Mockito.when;
|
import static org.mockito.Mockito.when;
|
||||||
|
import static org.mockito.Mockito.reset;
|
||||||
|
|
||||||
public class LingerMonitorTest extends TestCase {
|
public class LingerMonitorTest extends TestCase {
|
||||||
static final String CELLULAR = "CELLULAR";
|
static final String CELLULAR = "CELLULAR";
|
||||||
static final String WIFI = "WIFI";
|
static final String WIFI = "WIFI";
|
||||||
|
|
||||||
|
static final long LOW_RATE_LIMIT = DateUtils.MINUTE_IN_MILLIS;
|
||||||
|
static final long HIGH_RATE_LIMIT = 0;
|
||||||
|
|
||||||
|
static final int LOW_DAILY_LIMIT = 2;
|
||||||
|
static final int HIGH_DAILY_LIMIT = 1000;
|
||||||
|
|
||||||
LingerMonitor mMonitor;
|
LingerMonitor mMonitor;
|
||||||
|
|
||||||
@Mock ConnectivityService mConnService;
|
@Mock ConnectivityService mConnService;
|
||||||
@@ -59,7 +67,7 @@ public class LingerMonitorTest extends TestCase {
|
|||||||
when(mCtx.getPackageName()).thenReturn("com.android.server.connectivity");
|
when(mCtx.getPackageName()).thenReturn("com.android.server.connectivity");
|
||||||
when(mConnService.createNetworkMonitor(any(), any(), any(), any())).thenReturn(null);
|
when(mConnService.createNetworkMonitor(any(), any(), any(), any())).thenReturn(null);
|
||||||
|
|
||||||
mMonitor = new TestableLingerMonitor(mCtx, mNotifier);
|
mMonitor = new TestableLingerMonitor(mCtx, mNotifier, HIGH_DAILY_LIMIT, HIGH_RATE_LIMIT);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testTransitions() {
|
public void testTransitions() {
|
||||||
@@ -129,8 +137,78 @@ public class LingerMonitorTest extends TestCase {
|
|||||||
mMonitor.noteLingerDefaultNetwork(to, from);
|
mMonitor.noteLingerDefaultNetwork(to, from);
|
||||||
verify(mNotifier, times(1)).clearNotification(100);
|
verify(mNotifier, times(1)).clearNotification(100);
|
||||||
|
|
||||||
|
reset(mNotifier);
|
||||||
mMonitor.noteLingerDefaultNetwork(from, to);
|
mMonitor.noteLingerDefaultNetwork(from, to);
|
||||||
verifyToast(from, to);
|
verifyNoNotifications();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testMultipleNotifications() {
|
||||||
|
setNotificationSwitch(transition(WIFI, CELLULAR));
|
||||||
|
setNotificationType(LingerMonitor.NOTIFY_TYPE_NOTIFICATION);
|
||||||
|
NetworkAgentInfo wifi1 = wifiNai(100);
|
||||||
|
NetworkAgentInfo wifi2 = wifiNai(101);
|
||||||
|
NetworkAgentInfo cell = cellNai(102);
|
||||||
|
|
||||||
|
mMonitor.noteLingerDefaultNetwork(wifi1, cell);
|
||||||
|
verifyNotification(wifi1, cell);
|
||||||
|
|
||||||
|
mMonitor.noteLingerDefaultNetwork(cell, wifi2);
|
||||||
|
verify(mNotifier, times(1)).clearNotification(100);
|
||||||
|
|
||||||
|
reset(mNotifier);
|
||||||
|
mMonitor.noteLingerDefaultNetwork(wifi2, cell);
|
||||||
|
verifyNotification(wifi2, cell);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testRateLimiting() throws InterruptedException {
|
||||||
|
mMonitor = new TestableLingerMonitor(mCtx, mNotifier, HIGH_DAILY_LIMIT, LOW_RATE_LIMIT);
|
||||||
|
|
||||||
|
setNotificationSwitch(transition(WIFI, CELLULAR));
|
||||||
|
setNotificationType(LingerMonitor.NOTIFY_TYPE_NOTIFICATION);
|
||||||
|
NetworkAgentInfo wifi1 = wifiNai(100);
|
||||||
|
NetworkAgentInfo wifi2 = wifiNai(101);
|
||||||
|
NetworkAgentInfo wifi3 = wifiNai(102);
|
||||||
|
NetworkAgentInfo cell = cellNai(103);
|
||||||
|
|
||||||
|
mMonitor.noteLingerDefaultNetwork(wifi1, cell);
|
||||||
|
verifyNotification(wifi1, cell);
|
||||||
|
reset(mNotifier);
|
||||||
|
|
||||||
|
Thread.sleep(50);
|
||||||
|
mMonitor.noteLingerDefaultNetwork(cell, wifi2);
|
||||||
|
mMonitor.noteLingerDefaultNetwork(wifi2, cell);
|
||||||
|
verifyNoNotifications();
|
||||||
|
|
||||||
|
Thread.sleep(50);
|
||||||
|
mMonitor.noteLingerDefaultNetwork(cell, wifi3);
|
||||||
|
mMonitor.noteLingerDefaultNetwork(wifi3, cell);
|
||||||
|
verifyNoNotifications();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testDailyLimiting() throws InterruptedException {
|
||||||
|
mMonitor = new TestableLingerMonitor(mCtx, mNotifier, LOW_DAILY_LIMIT, HIGH_RATE_LIMIT);
|
||||||
|
|
||||||
|
setNotificationSwitch(transition(WIFI, CELLULAR));
|
||||||
|
setNotificationType(LingerMonitor.NOTIFY_TYPE_NOTIFICATION);
|
||||||
|
NetworkAgentInfo wifi1 = wifiNai(100);
|
||||||
|
NetworkAgentInfo wifi2 = wifiNai(101);
|
||||||
|
NetworkAgentInfo wifi3 = wifiNai(102);
|
||||||
|
NetworkAgentInfo cell = cellNai(103);
|
||||||
|
|
||||||
|
mMonitor.noteLingerDefaultNetwork(wifi1, cell);
|
||||||
|
verifyNotification(wifi1, cell);
|
||||||
|
reset(mNotifier);
|
||||||
|
|
||||||
|
Thread.sleep(50);
|
||||||
|
mMonitor.noteLingerDefaultNetwork(cell, wifi2);
|
||||||
|
mMonitor.noteLingerDefaultNetwork(wifi2, cell);
|
||||||
|
verifyNotification(wifi2, cell);
|
||||||
|
reset(mNotifier);
|
||||||
|
|
||||||
|
Thread.sleep(50);
|
||||||
|
mMonitor.noteLingerDefaultNetwork(cell, wifi3);
|
||||||
|
mMonitor.noteLingerDefaultNetwork(wifi3, cell);
|
||||||
|
verifyNoNotifications();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testUniqueNotification() {
|
public void testUniqueNotification() {
|
||||||
@@ -149,7 +227,7 @@ public class LingerMonitorTest extends TestCase {
|
|||||||
verifyNotification(from, to);
|
verifyNotification(from, to);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testIgnoreUnvalidatedNetworks() {
|
public void testIgnoreNeverValidatedNetworks() {
|
||||||
setNotificationType(LingerMonitor.NOTIFY_TYPE_TOAST);
|
setNotificationType(LingerMonitor.NOTIFY_TYPE_TOAST);
|
||||||
setNotificationSwitch(transition(WIFI, CELLULAR));
|
setNotificationSwitch(transition(WIFI, CELLULAR));
|
||||||
NetworkAgentInfo from = wifiNai(100);
|
NetworkAgentInfo from = wifiNai(100);
|
||||||
@@ -160,6 +238,17 @@ public class LingerMonitorTest extends TestCase {
|
|||||||
verifyNoNotifications();
|
verifyNoNotifications();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void testIgnoreCurrentlyValidatedNetworks() {
|
||||||
|
setNotificationType(LingerMonitor.NOTIFY_TYPE_TOAST);
|
||||||
|
setNotificationSwitch(transition(WIFI, CELLULAR));
|
||||||
|
NetworkAgentInfo from = wifiNai(100);
|
||||||
|
NetworkAgentInfo to = cellNai(101);
|
||||||
|
from.lastValidated = true;
|
||||||
|
|
||||||
|
mMonitor.noteLingerDefaultNetwork(from, to);
|
||||||
|
verifyNoNotifications();
|
||||||
|
}
|
||||||
|
|
||||||
public void testNoNotificationType() {
|
public void testNoNotificationType() {
|
||||||
setNotificationType(LingerMonitor.NOTIFY_TYPE_TOAST);
|
setNotificationType(LingerMonitor.NOTIFY_TYPE_TOAST);
|
||||||
setNotificationSwitch();
|
setNotificationSwitch();
|
||||||
@@ -215,7 +304,6 @@ public class LingerMonitorTest extends TestCase {
|
|||||||
void verifyNoNotifications() {
|
void verifyNoNotifications() {
|
||||||
verifyNoToast();
|
verifyNoToast();
|
||||||
verifyNoNotification();
|
verifyNoNotification();
|
||||||
verify(mNotifier, never()).clearNotification(anyInt());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void verifyToast(NetworkAgentInfo from, NetworkAgentInfo to) {
|
void verifyToast(NetworkAgentInfo from, NetworkAgentInfo to) {
|
||||||
@@ -251,8 +339,8 @@ public class LingerMonitorTest extends TestCase {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public static class TestableLingerMonitor extends LingerMonitor {
|
public static class TestableLingerMonitor extends LingerMonitor {
|
||||||
public TestableLingerMonitor(Context c, NetworkNotificationManager n) {
|
public TestableLingerMonitor(Context c, NetworkNotificationManager n, int l, long r) {
|
||||||
super(c, n);
|
super(c, n, l, r);
|
||||||
}
|
}
|
||||||
@Override protected PendingIntent createNotificationIntent() {
|
@Override protected PendingIntent createNotificationIntent() {
|
||||||
return null;
|
return null;
|
||||||
|
|||||||
Reference in New Issue
Block a user