使用NetworkStatsManager获取移动数据使用logging

我想知道数据使用的历史,并注意到“新”android-6 NetworkStatsManager这似乎是积极的(我已经使用TrafficStats一段时间,但不会涵盖任何重启前)。

从API文档:

注意:此API需要权限PACKAGE_USAGE_STATS,这是系统级权限,不会授予第三方应用程序。 但是,声明权限意味着使用API​​的意图,并且设备的用户可以通过设置应用程序授予权限。 configuration文件所有者应用程序被自动授予查询他们pipe理的configuration文件(即除querySummaryForDevice(int,String,long,long)之外的任何查询)数据的权限。 设备所有者应用同样可以访问主用户的使用数据。

我想知道聚合级别的数据使用情况,而不是使用数据的应用程序,所以我尝试像这样使用它:

 NetworkStatsManager service = context.getSystemService(NetworkStatsManager.class); NetworkStats.Bucket bucket = service.querySummaryForDevice(ConnectivityManager.TYPE_MOBILE, null, from, to); ... 

不幸的是,抛出一个SecurityException

 java.lang.SecurityException: NetworkStats: Neither user 10174 nor current process has android.permission.READ_NETWORK_USAGE_HISTORY. at android.os.Parcel.readException(Parcel.java:1620) at android.os.Parcel.readException(Parcel.java:1573) at android.net.INetworkStatsSession$Stub$Proxy.getDeviceSummaryForNetwork(INetworkStatsSession.java:259) at android.app.usage.NetworkStats.getDeviceSummaryForNetwork(NetworkStats.java:316) at android.app.usage.NetworkStatsManager.querySummaryForDevice(NetworkStatsManager.java:100) ... 

第三方应用程序不允许使用android.permission.READ_NETWORK_USAGE_HISTORY权限。 所以这似乎是一个死胡同。

不过,我钻了一下内部,发现你可以使用内部/隐藏的API做同样的事情,而不要求任何权限:

 INetworkStatsService service = INetworkStatsService.Stub.asInterface( ServiceManager.getService(Context.NETWORK_STATS_SERVICE)); INetworkStatsSession session = service.openSession(); NetworkTemplate mobileTemplate = NetworkTemplate.buildTemplateMobileWildcard(); int fields = NetworkStatsHistory.FIELD_RX_BYTES | NetworkStatsHistory.FIELD_TX_BYTES; NetworkStatsHistory mobileHistory = session.getHistoryForNetwork(mobileTemplate, fields); for (int i = 0; i < mobileHistory.size(); i++) { NetworkStatsHistory.Entry entry = mobileHistory.getValues(i, null); ... } session.close(); 

我真的想做公共API,所以,我该怎么做?

Solutions Collecting From Web of "使用NetworkStatsManager获取移动数据使用logging"

有办法获得对NetworkStateManager的访问而不访问私有API。 这里是步骤:

  1. AndroidManifest.xml声明所需的权限:

     <uses-permission android:name="android.permission.READ_PHONE_STATE"/> <uses-permission android:name="android.permission.PACKAGE_USAGE_STATS" tools:ignore="ProtectedPermissions"/> 
    1. Activity询问权限

    android.permission.PACKAGE_USAGE_STATS不是一个正常的权限,不能简单的请求。 为了检查,权限是否被授予,检查:

     AppOpsManager appOps = (AppOpsManager) getSystemService(Context.APP_OPS_SERVICE); int mode = appOps.checkOpNoThrow(AppOpsManager.OPSTR_GET_USAGE_STATS, android.os.Process.myUid(), getPackageName()); if (mode == AppOpsManager.MODE_ALLOWED) { return true; } 

    要请求这个权限,只需调用Intent

     Intent intent = new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS); startActivity(intent); 

    还需要另一个权限: Manifest.permission.READ_PHONE_STATE 。 只有在需要获取移动数据统计信息时才需要。 但是,这是正常的许可,所以可以作为任何其他许可请求

    1. 使用NetworkStatsManager

做了一个示例Github回购示范使用。

 /* getting youtube usage for both mobile and wifi. */ public long getYoutubeTotalusage(Context context) { String subId = getSubscriberId(context, ConnectivityManager.TYPE_MOBILE); return getYoutubeUsage(ConnectivityManager.TYPE_MOBILE, subId) + getYoutubeUsage(ConnectivityManager.TYPE_WIFI, ""); } private long getYoutubeUsage(int networkType, String subScriberId) { NetworkStats networkStatsByApp; long currentYoutubeUsage = 0L; try { networkStatsByApp = networkStatsManager.querySummary(networkType, subScriberId, 0, System.currentTimeMillis()); do { NetworkStats.Bucket bucket = new NetworkStats.Bucket(); networkStatsByApp.getNextBucket(bucket); if (bucket.getUid() == packageUid) { //rajeesh : in some devices this is immediately looping twice and the second iteration is returning correct value. So result returning is moved to the end. currentYoutubeUsage = (bucket.getRxBytes() + bucket.getTxBytes()); } } while (networkStatsByApp.hasNextBucket()); } catch (RemoteException e) { e.printStackTrace(); } return currentYoutubeUsage; } private String getSubscriberId(Context context, int networkType) { if (ConnectivityManager.TYPE_MOBILE == networkType) { TelephonyManager tm = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE); return tm.getSubscriberId(); } return ""; }