Use the new network stats loader to get usage data for all apps.
- change to use the new NetworkStats.Bucket instead of NetworkStats.Entry when iterating through the detail data. Bug: 111751694 Test: make RunSettingsRoboTests Change-Id: I305cc384320e4a72531d80dd9a00a3034ab12837
This commit is contained in:
@@ -21,6 +21,8 @@ import static android.net.TrafficStats.UID_TETHERING;
|
|||||||
import static android.telephony.TelephonyManager.SIM_STATE_READY;
|
import static android.telephony.TelephonyManager.SIM_STATE_READY;
|
||||||
|
|
||||||
import android.app.ActivityManager;
|
import android.app.ActivityManager;
|
||||||
|
import android.app.usage.NetworkStats;
|
||||||
|
import android.app.usage.NetworkStats.Bucket;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.pm.UserInfo;
|
import android.content.pm.UserInfo;
|
||||||
@@ -28,7 +30,6 @@ import android.graphics.Color;
|
|||||||
import android.net.ConnectivityManager;
|
import android.net.ConnectivityManager;
|
||||||
import android.net.INetworkStatsSession;
|
import android.net.INetworkStatsSession;
|
||||||
import android.net.NetworkPolicy;
|
import android.net.NetworkPolicy;
|
||||||
import android.net.NetworkStats;
|
|
||||||
import android.net.NetworkStatsHistory;
|
import android.net.NetworkStatsHistory;
|
||||||
import android.net.NetworkTemplate;
|
import android.net.NetworkTemplate;
|
||||||
import android.net.TrafficStats;
|
import android.net.TrafficStats;
|
||||||
@@ -65,7 +66,7 @@ import com.android.settings.widget.LoadingViewController;
|
|||||||
import com.android.settingslib.AppItem;
|
import com.android.settingslib.AppItem;
|
||||||
import com.android.settingslib.net.ChartData;
|
import com.android.settingslib.net.ChartData;
|
||||||
import com.android.settingslib.net.ChartDataLoaderCompat;
|
import com.android.settingslib.net.ChartDataLoaderCompat;
|
||||||
import com.android.settingslib.net.SummaryForAllUidLoaderCompat;
|
import com.android.settingslib.net.NetworkStatsDetailLoader;
|
||||||
import com.android.settingslib.net.UidDetailProvider;
|
import com.android.settingslib.net.UidDetailProvider;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
@@ -78,8 +79,9 @@ import java.util.List;
|
|||||||
*/
|
*/
|
||||||
public class DataUsageListV2 extends DataUsageBaseFragment {
|
public class DataUsageListV2 extends DataUsageBaseFragment {
|
||||||
|
|
||||||
public static final String EXTRA_SUB_ID = "sub_id";
|
static final String EXTRA_SUB_ID = "sub_id";
|
||||||
public static final String EXTRA_NETWORK_TEMPLATE = "network_template";
|
static final String EXTRA_NETWORK_TEMPLATE = "network_template";
|
||||||
|
static final String EXTRA_NETWORK_TYPE = "network_type";
|
||||||
|
|
||||||
private static final String TAG = "DataUsageListV2";
|
private static final String TAG = "DataUsageListV2";
|
||||||
private static final boolean LOGD = false;
|
private static final boolean LOGD = false;
|
||||||
@@ -106,6 +108,8 @@ public class DataUsageListV2 extends DataUsageBaseFragment {
|
|||||||
NetworkTemplate mTemplate;
|
NetworkTemplate mTemplate;
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
int mSubId = SubscriptionManager.INVALID_SUBSCRIPTION_ID;
|
int mSubId = SubscriptionManager.INVALID_SUBSCRIPTION_ID;
|
||||||
|
@VisibleForTesting
|
||||||
|
int mNetworkType;
|
||||||
private ChartData mChartData;
|
private ChartData mChartData;
|
||||||
|
|
||||||
private LoadingViewController mLoadingViewController;
|
private LoadingViewController mLoadingViewController;
|
||||||
@@ -249,6 +253,7 @@ public class DataUsageListV2 extends DataUsageBaseFragment {
|
|||||||
if (args != null) {
|
if (args != null) {
|
||||||
mSubId = args.getInt(EXTRA_SUB_ID, SubscriptionManager.INVALID_SUBSCRIPTION_ID);
|
mSubId = args.getInt(EXTRA_SUB_ID, SubscriptionManager.INVALID_SUBSCRIPTION_ID);
|
||||||
mTemplate = args.getParcelable(EXTRA_NETWORK_TEMPLATE);
|
mTemplate = args.getParcelable(EXTRA_NETWORK_TEMPLATE);
|
||||||
|
mNetworkType = args.getInt(EXTRA_NETWORK_TYPE, ConnectivityManager.TYPE_MOBILE);
|
||||||
}
|
}
|
||||||
if (mTemplate == null && mSubId == SubscriptionManager.INVALID_SUBSCRIPTION_ID) {
|
if (mTemplate == null && mSubId == SubscriptionManager.INVALID_SUBSCRIPTION_ID) {
|
||||||
final Intent intent = getIntent();
|
final Intent intent = getIntent();
|
||||||
@@ -336,8 +341,8 @@ public class DataUsageListV2 extends DataUsageBaseFragment {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// kick off loader for detailed stats
|
// kick off loader for detailed stats
|
||||||
getLoaderManager().restartLoader(LOADER_SUMMARY,
|
getLoaderManager().restartLoader(LOADER_SUMMARY, null /* args */,
|
||||||
SummaryForAllUidLoaderCompat.buildArgs(mTemplate, start, end), mSummaryCallbacks);
|
mNetworkStatsDetailCallbacks);
|
||||||
|
|
||||||
final long totalBytes = entry != null ? entry.rxBytes + entry.txBytes : 0;
|
final long totalBytes = entry != null ? entry.rxBytes + entry.txBytes : 0;
|
||||||
final CharSequence totalPhrase = DataUsageUtils.formatDataUsage(context, totalBytes);
|
final CharSequence totalPhrase = DataUsageUtils.formatDataUsage(context, totalBytes);
|
||||||
@@ -348,22 +353,26 @@ public class DataUsageListV2 extends DataUsageBaseFragment {
|
|||||||
* Bind the given {@link NetworkStats}, or {@code null} to clear list.
|
* Bind the given {@link NetworkStats}, or {@code null} to clear list.
|
||||||
*/
|
*/
|
||||||
public void bindStats(NetworkStats stats, int[] restrictedUids) {
|
public void bindStats(NetworkStats stats, int[] restrictedUids) {
|
||||||
ArrayList<AppItem> items = new ArrayList<>();
|
mApps.removeAll();
|
||||||
|
if (stats == null) {
|
||||||
|
if (LOGD) {
|
||||||
|
Log.d(TAG, "No network stats data. App list cleared.");
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final ArrayList<AppItem> items = new ArrayList<>();
|
||||||
long largest = 0;
|
long largest = 0;
|
||||||
|
|
||||||
final int currentUserId = ActivityManager.getCurrentUser();
|
final int currentUserId = ActivityManager.getCurrentUser();
|
||||||
UserManager userManager = UserManager.get(getContext());
|
final UserManager userManager = UserManager.get(getContext());
|
||||||
final List<UserHandle> profiles = userManager.getUserProfiles();
|
final List<UserHandle> profiles = userManager.getUserProfiles();
|
||||||
final SparseArray<AppItem> knownItems = new SparseArray<AppItem>();
|
final SparseArray<AppItem> knownItems = new SparseArray<AppItem>();
|
||||||
|
|
||||||
NetworkStats.Entry entry = null;
|
final Bucket bucket = new Bucket();
|
||||||
final int size = stats != null ? stats.size() : 0;
|
while (stats.hasNextBucket() && stats.getNextBucket(bucket)) {
|
||||||
for (int i = 0; i < size; i++) {
|
|
||||||
entry = stats.getValues(i, entry);
|
|
||||||
|
|
||||||
// Decide how to collapse items together
|
// Decide how to collapse items together
|
||||||
final int uid = entry.uid;
|
final int uid = bucket.getUid();
|
||||||
|
|
||||||
final int collapseKey;
|
final int collapseKey;
|
||||||
final int category;
|
final int category;
|
||||||
final int userId = UserHandle.getUserId(uid);
|
final int userId = UserHandle.getUserId(uid);
|
||||||
@@ -372,8 +381,8 @@ public class DataUsageListV2 extends DataUsageBaseFragment {
|
|||||||
if (userId != currentUserId) {
|
if (userId != currentUserId) {
|
||||||
// Add to a managed user item.
|
// Add to a managed user item.
|
||||||
final int managedKey = UidDetailProvider.buildKeyForUser(userId);
|
final int managedKey = UidDetailProvider.buildKeyForUser(userId);
|
||||||
largest = accumulate(managedKey, knownItems, entry, AppItem.CATEGORY_USER,
|
largest = accumulate(managedKey, knownItems, bucket,
|
||||||
items, largest);
|
AppItem.CATEGORY_USER, items, largest);
|
||||||
}
|
}
|
||||||
// Add to app item.
|
// Add to app item.
|
||||||
collapseKey = uid;
|
collapseKey = uid;
|
||||||
@@ -397,8 +406,9 @@ public class DataUsageListV2 extends DataUsageBaseFragment {
|
|||||||
collapseKey = android.os.Process.SYSTEM_UID;
|
collapseKey = android.os.Process.SYSTEM_UID;
|
||||||
category = AppItem.CATEGORY_APP;
|
category = AppItem.CATEGORY_APP;
|
||||||
}
|
}
|
||||||
largest = accumulate(collapseKey, knownItems, entry, category, items, largest);
|
largest = accumulate(collapseKey, knownItems, bucket, category, items, largest);
|
||||||
}
|
}
|
||||||
|
stats.close();
|
||||||
|
|
||||||
final int restrictedUidsMax = restrictedUids.length;
|
final int restrictedUidsMax = restrictedUids.length;
|
||||||
for (int i = 0; i < restrictedUidsMax; ++i) {
|
for (int i = 0; i < restrictedUidsMax; ++i) {
|
||||||
@@ -419,7 +429,6 @@ public class DataUsageListV2 extends DataUsageBaseFragment {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Collections.sort(items);
|
Collections.sort(items);
|
||||||
mApps.removeAll();
|
|
||||||
for (int i = 0; i < items.size(); i++) {
|
for (int i = 0; i < items.size(); i++) {
|
||||||
final int percentTotal = largest != 0 ? (int) (items.get(i).total * 100 / largest) : 0;
|
final int percentTotal = largest != 0 ? (int) (items.get(i).total * 100 / largest) : 0;
|
||||||
AppDataUsagePreference preference = new AppDataUsagePreference(getContext(),
|
AppDataUsagePreference preference = new AppDataUsagePreference(getContext(),
|
||||||
@@ -456,12 +465,12 @@ public class DataUsageListV2 extends DataUsageBaseFragment {
|
|||||||
*
|
*
|
||||||
* @param collapseKey the collapse key used to map the item.
|
* @param collapseKey the collapse key used to map the item.
|
||||||
* @param knownItems collection of known (already existing) items.
|
* @param knownItems collection of known (already existing) items.
|
||||||
* @param entry the network stats entry to extract data usage from.
|
* @param bucket the network stats bucket to extract data usage from.
|
||||||
* @param itemCategory the item is categorized on the list view by this category. Must be
|
* @param itemCategory the item is categorized on the list view by this category. Must be
|
||||||
*/
|
*/
|
||||||
private static long accumulate(int collapseKey, final SparseArray<AppItem> knownItems,
|
private static long accumulate(int collapseKey, final SparseArray<AppItem> knownItems,
|
||||||
NetworkStats.Entry entry, int itemCategory, ArrayList<AppItem> items, long largest) {
|
Bucket bucket, int itemCategory, ArrayList<AppItem> items, long largest) {
|
||||||
final int uid = entry.uid;
|
final int uid = bucket.getUid();
|
||||||
AppItem item = knownItems.get(collapseKey);
|
AppItem item = knownItems.get(collapseKey);
|
||||||
if (item == null) {
|
if (item == null) {
|
||||||
item = new AppItem(collapseKey);
|
item = new AppItem(collapseKey);
|
||||||
@@ -470,7 +479,7 @@ public class DataUsageListV2 extends DataUsageBaseFragment {
|
|||||||
knownItems.put(item.key, item);
|
knownItems.put(item.key, item);
|
||||||
}
|
}
|
||||||
item.addUid(uid);
|
item.addUid(uid);
|
||||||
item.total += entry.rxBytes + entry.txBytes;
|
item.total += bucket.getRxBytes() + bucket.getTxBytes();
|
||||||
return Math.max(largest, item.total);
|
return Math.max(largest, item.total);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -579,11 +588,16 @@ public class DataUsageListV2 extends DataUsageBaseFragment {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
private final LoaderCallbacks<NetworkStats> mSummaryCallbacks = new LoaderCallbacks<
|
private final LoaderCallbacks<NetworkStats> mNetworkStatsDetailCallbacks =
|
||||||
NetworkStats>() {
|
new LoaderCallbacks<NetworkStats>() {
|
||||||
@Override
|
@Override
|
||||||
public Loader<NetworkStats> onCreateLoader(int id, Bundle args) {
|
public Loader<NetworkStats> onCreateLoader(int id, Bundle args) {
|
||||||
return new SummaryForAllUidLoaderCompat(getActivity(), mStatsSession, args);
|
return new NetworkStatsDetailLoader.Builder(getContext())
|
||||||
|
.setStartTime(mChart.getInspectStart())
|
||||||
|
.setEndTime(mChart.getInspectEnd())
|
||||||
|
.setNetworkType(mNetworkType)
|
||||||
|
.setSubscriptionId(mSubId)
|
||||||
|
.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@@ -17,6 +17,7 @@ package com.android.settings.datausage;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.res.TypedArray;
|
import android.content.res.TypedArray;
|
||||||
|
import android.net.ConnectivityManager;
|
||||||
import android.net.NetworkTemplate;
|
import android.net.NetworkTemplate;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.util.AttributeSet;
|
import android.util.AttributeSet;
|
||||||
@@ -81,6 +82,8 @@ public class DataUsagePreference extends Preference implements TemplatePreferenc
|
|||||||
if (FeatureFlagUtils.isEnabled(getContext(), FeatureFlags.DATA_USAGE_V2)) {
|
if (FeatureFlagUtils.isEnabled(getContext(), FeatureFlags.DATA_USAGE_V2)) {
|
||||||
args.putParcelable(DataUsageListV2.EXTRA_NETWORK_TEMPLATE, mTemplate);
|
args.putParcelable(DataUsageListV2.EXTRA_NETWORK_TEMPLATE, mTemplate);
|
||||||
args.putInt(DataUsageListV2.EXTRA_SUB_ID, mSubId);
|
args.putInt(DataUsageListV2.EXTRA_SUB_ID, mSubId);
|
||||||
|
args.putInt(DataUsageListV2.EXTRA_NETWORK_TYPE, mTemplate.isMatchRuleMobile()
|
||||||
|
? ConnectivityManager.TYPE_MOBILE : ConnectivityManager.TYPE_WIFI);
|
||||||
launcher = new SubSettingLauncher(getContext())
|
launcher = new SubSettingLauncher(getContext())
|
||||||
.setArguments(args)
|
.setArguments(args)
|
||||||
.setDestination(DataUsageListV2.class.getName())
|
.setDestination(DataUsageListV2.class.getName())
|
||||||
|
@@ -26,6 +26,7 @@ import static org.mockito.Mockito.when;
|
|||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
|
import android.net.ConnectivityManager;
|
||||||
import android.net.NetworkTemplate;
|
import android.net.NetworkTemplate;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.provider.Settings;
|
import android.provider.Settings;
|
||||||
@@ -97,6 +98,18 @@ public class DataUsageListV2Test {
|
|||||||
assertThat(mDataUsageList.mSubId).isEqualTo(3);
|
assertThat(mDataUsageList.mSubId).isEqualTo(3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void processArgument_shouldGetNetworkTypeFromArgument() {
|
||||||
|
final Bundle args = new Bundle();
|
||||||
|
args.putInt(DataUsageListV2.EXTRA_NETWORK_TYPE, ConnectivityManager.TYPE_WIFI);
|
||||||
|
args.putInt(DataUsageListV2.EXTRA_SUB_ID, 3);
|
||||||
|
mDataUsageList.setArguments(args);
|
||||||
|
|
||||||
|
mDataUsageList.processArgument();
|
||||||
|
|
||||||
|
assertThat(mDataUsageList.mNetworkType).isEqualTo(ConnectivityManager.TYPE_WIFI);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void processArgument_fromIntent_shouldGetTemplateFromIntent() {
|
public void processArgument_fromIntent_shouldGetTemplateFromIntent() {
|
||||||
final FragmentActivity activity = mock(FragmentActivity.class);
|
final FragmentActivity activity = mock(FragmentActivity.class);
|
||||||
|
Reference in New Issue
Block a user