1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.settingslib.net;
18 
19 import static android.net.ConnectivityManager.TYPE_MOBILE;
20 import static android.net.NetworkStatsHistory.FIELD_RX_BYTES;
21 import static android.net.NetworkStatsHistory.FIELD_TX_BYTES;
22 import static android.net.TrafficStats.MB_IN_BYTES;
23 import static android.telephony.TelephonyManager.SIM_STATE_READY;
24 import static android.text.format.DateUtils.FORMAT_ABBREV_MONTH;
25 import static android.text.format.DateUtils.FORMAT_SHOW_DATE;
26 
27 import android.app.usage.NetworkStats.Bucket;
28 import android.app.usage.NetworkStatsManager;
29 import android.content.Context;
30 import android.net.ConnectivityManager;
31 import android.net.INetworkStatsService;
32 import android.net.INetworkStatsSession;
33 import android.net.NetworkPolicy;
34 import android.net.NetworkPolicyManager;
35 import android.net.NetworkTemplate;
36 import android.os.RemoteException;
37 import android.os.ServiceManager;
38 import android.telephony.SubscriptionManager;
39 import android.telephony.TelephonyManager;
40 import android.text.format.DateUtils;
41 import android.util.Log;
42 import android.util.Range;
43 
44 import com.android.internal.R;
45 import com.android.internal.annotations.VisibleForTesting;
46 import com.android.internal.util.ArrayUtils;
47 
48 import java.time.ZonedDateTime;
49 import java.util.Iterator;
50 import java.util.Locale;
51 
52 public class DataUsageController {
53 
54     private static final String TAG = "DataUsageController";
55     private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG);
56     private static final int FIELDS = FIELD_RX_BYTES | FIELD_TX_BYTES;
57     private static final StringBuilder PERIOD_BUILDER = new StringBuilder(50);
58     private static final java.util.Formatter PERIOD_FORMATTER = new java.util.Formatter(
59             PERIOD_BUILDER, Locale.getDefault());
60 
61     private final Context mContext;
62     private final ConnectivityManager mConnectivityManager;
63     private final INetworkStatsService mStatsService;
64     private final NetworkPolicyManager mPolicyManager;
65     private final NetworkStatsManager mNetworkStatsManager;
66 
67     private INetworkStatsSession mSession;
68     private Callback mCallback;
69     private NetworkNameProvider mNetworkController;
70     private int mSubscriptionId;
71 
DataUsageController(Context context)72     public DataUsageController(Context context) {
73         mContext = context;
74         mConnectivityManager = ConnectivityManager.from(context);
75         mStatsService = INetworkStatsService.Stub.asInterface(
76                 ServiceManager.getService(Context.NETWORK_STATS_SERVICE));
77         mPolicyManager = NetworkPolicyManager.from(mContext);
78         mNetworkStatsManager = context.getSystemService(NetworkStatsManager.class);
79         mSubscriptionId = SubscriptionManager.INVALID_SUBSCRIPTION_ID;
80     }
81 
setNetworkController(NetworkNameProvider networkController)82     public void setNetworkController(NetworkNameProvider networkController) {
83         mNetworkController = networkController;
84     }
85 
86     /**
87      * By default this class will just get data usage information for the default data subscription,
88      * but this method can be called to require it to use an explicit subscription id which may be
89      * different from the default one (this is useful for the case of multi-SIM devices).
90      */
setSubscriptionId(int subscriptionId)91     public void setSubscriptionId(int subscriptionId) {
92         mSubscriptionId = subscriptionId;
93     }
94 
95     /**
96      * Returns the default warning level in bytes.
97      */
getDefaultWarningLevel()98     public long getDefaultWarningLevel() {
99         return MB_IN_BYTES
100                 * mContext.getResources().getInteger(R.integer.default_data_warning_level_mb);
101     }
102 
setCallback(Callback callback)103     public void setCallback(Callback callback) {
104         mCallback = callback;
105     }
106 
warn(String msg)107     private DataUsageInfo warn(String msg) {
108         Log.w(TAG, "Failed to get data usage, " + msg);
109         return null;
110     }
111 
getDataUsageInfo()112     public DataUsageInfo getDataUsageInfo() {
113         NetworkTemplate template = DataUsageUtils.getMobileTemplate(mContext, mSubscriptionId);
114 
115         return getDataUsageInfo(template);
116     }
117 
getWifiDataUsageInfo()118     public DataUsageInfo getWifiDataUsageInfo() {
119         NetworkTemplate template = NetworkTemplate.buildTemplateWifiWildcard();
120         return getDataUsageInfo(template);
121     }
122 
getDataUsageInfo(NetworkTemplate template)123     public DataUsageInfo getDataUsageInfo(NetworkTemplate template) {
124         final NetworkPolicy policy = findNetworkPolicy(template);
125         final long now = System.currentTimeMillis();
126         final long start, end;
127         final Iterator<Range<ZonedDateTime>> it = (policy != null) ? policy.cycleIterator() : null;
128         if (it != null && it.hasNext()) {
129             final Range<ZonedDateTime> cycle = it.next();
130             start = cycle.getLower().toInstant().toEpochMilli();
131             end = cycle.getUpper().toInstant().toEpochMilli();
132         } else {
133             // period = last 4 wks
134             end = now;
135             start = now - DateUtils.WEEK_IN_MILLIS * 4;
136         }
137         final long totalBytes = getUsageLevel(template, start, end);
138         if (totalBytes < 0L) {
139             return warn("no entry data");
140         }
141         final DataUsageInfo usage = new DataUsageInfo();
142         usage.startDate = start;
143         usage.usageLevel = totalBytes;
144         usage.period = formatDateRange(start, end);
145         usage.cycleStart = start;
146         usage.cycleEnd = end;
147 
148         if (policy != null) {
149             usage.limitLevel = policy.limitBytes > 0 ? policy.limitBytes : 0;
150             usage.warningLevel = policy.warningBytes > 0 ? policy.warningBytes : 0;
151         } else {
152             usage.warningLevel = getDefaultWarningLevel();
153         }
154         if (usage != null && mNetworkController != null) {
155             usage.carrier = mNetworkController.getMobileDataNetworkName();
156         }
157         return usage;
158     }
159 
160     /**
161      * Get the total usage level recorded in the network history
162      * @param template the network template to retrieve the network history
163      * @return the total usage level recorded in the network history or -1L if there is error
164      * retrieving the data.
165      */
getHistoricalUsageLevel(NetworkTemplate template)166     public long getHistoricalUsageLevel(NetworkTemplate template) {
167         return getUsageLevel(template, 0L /* start */, System.currentTimeMillis() /* end */);
168     }
169 
getUsageLevel(NetworkTemplate template, long start, long end)170     private long getUsageLevel(NetworkTemplate template, long start, long end) {
171         try {
172             final Bucket bucket = mNetworkStatsManager.querySummaryForDevice(template, start, end);
173             if (bucket != null) {
174                 return bucket.getRxBytes() + bucket.getTxBytes();
175             }
176             Log.w(TAG, "Failed to get data usage, no entry data");
177         } catch (RemoteException e) {
178             Log.w(TAG, "Failed to get data usage, remote call failed");
179         }
180         return -1L;
181     }
182 
findNetworkPolicy(NetworkTemplate template)183     private NetworkPolicy findNetworkPolicy(NetworkTemplate template) {
184         if (mPolicyManager == null || template == null) return null;
185         final NetworkPolicy[] policies = mPolicyManager.getNetworkPolicies();
186         if (policies == null) return null;
187         final int N = policies.length;
188         for (int i = 0; i < N; i++) {
189             final NetworkPolicy policy = policies[i];
190             if (policy != null && template.equals(policy.template)) {
191                 return policy;
192             }
193         }
194         return null;
195     }
196 
statsBucketToString(Bucket bucket)197     private static String statsBucketToString(Bucket bucket) {
198         return bucket == null ? null : new StringBuilder("Entry[")
199             .append("bucketDuration=").append(bucket.getEndTimeStamp() - bucket.getStartTimeStamp())
200             .append(",bucketStart=").append(bucket.getStartTimeStamp())
201             .append(",rxBytes=").append(bucket.getRxBytes())
202             .append(",rxPackets=").append(bucket.getRxPackets())
203             .append(",txBytes=").append(bucket.getTxBytes())
204             .append(",txPackets=").append(bucket.getTxPackets())
205             .append(']').toString();
206     }
207 
208     @VisibleForTesting
getTelephonyManager()209     public TelephonyManager getTelephonyManager() {
210         int subscriptionId = mSubscriptionId;
211 
212         // If mSubscriptionId is invalid, get default data sub.
213         if (!SubscriptionManager.isValidSubscriptionId(subscriptionId)) {
214             subscriptionId = SubscriptionManager.getDefaultDataSubscriptionId();
215         }
216 
217         // If data sub is also invalid, get any active sub.
218         if (!SubscriptionManager.isValidSubscriptionId(subscriptionId)) {
219             int[] activeSubIds = SubscriptionManager.from(mContext).getActiveSubscriptionIdList();
220             if (!ArrayUtils.isEmpty(activeSubIds)) {
221                 subscriptionId = activeSubIds[0];
222             }
223         }
224 
225         return TelephonyManager.from(mContext).createForSubscriptionId(subscriptionId);
226     }
227 
setMobileDataEnabled(boolean enabled)228     public void setMobileDataEnabled(boolean enabled) {
229         Log.d(TAG, "setMobileDataEnabled: enabled=" + enabled);
230         getTelephonyManager().setDataEnabled(enabled);
231         if (mCallback != null) {
232             mCallback.onMobileDataEnabled(enabled);
233         }
234     }
235 
isMobileDataSupported()236     public boolean isMobileDataSupported() {
237         // require both supported network and ready SIM
238         return mConnectivityManager.isNetworkSupported(TYPE_MOBILE)
239                 && getTelephonyManager().getSimState() == SIM_STATE_READY;
240     }
241 
isMobileDataEnabled()242     public boolean isMobileDataEnabled() {
243         return getTelephonyManager().isDataEnabled();
244     }
245 
getNetworkType(NetworkTemplate networkTemplate)246     static int getNetworkType(NetworkTemplate networkTemplate) {
247         if (networkTemplate == null) {
248             return ConnectivityManager.TYPE_NONE;
249         }
250         final int matchRule = networkTemplate.getMatchRule();
251         switch (matchRule) {
252             case NetworkTemplate.MATCH_MOBILE:
253             case NetworkTemplate.MATCH_MOBILE_WILDCARD:
254                 return ConnectivityManager.TYPE_MOBILE;
255             case NetworkTemplate.MATCH_WIFI:
256             case NetworkTemplate.MATCH_WIFI_WILDCARD:
257                 return  ConnectivityManager.TYPE_WIFI;
258             case NetworkTemplate.MATCH_ETHERNET:
259                 return  ConnectivityManager.TYPE_ETHERNET;
260             default:
261                 return ConnectivityManager.TYPE_MOBILE;
262         }
263     }
264 
getActiveSubscriberId()265     private String getActiveSubscriberId() {
266         final String actualSubscriberId = getTelephonyManager().getSubscriberId();
267         return actualSubscriberId;
268     }
269 
formatDateRange(long start, long end)270     private String formatDateRange(long start, long end) {
271         final int flags = FORMAT_SHOW_DATE | FORMAT_ABBREV_MONTH;
272         synchronized (PERIOD_BUILDER) {
273             PERIOD_BUILDER.setLength(0);
274             return DateUtils.formatDateRange(mContext, PERIOD_FORMATTER, start, end, flags, null)
275                     .toString();
276         }
277     }
278 
279     public interface NetworkNameProvider {
getMobileDataNetworkName()280         String getMobileDataNetworkName();
281     }
282 
283     public static class DataUsageInfo {
284         public String carrier;
285         public String period;
286         public long startDate;
287         public long limitLevel;
288         public long warningLevel;
289         public long usageLevel;
290         public long cycleStart;
291         public long cycleEnd;
292     }
293 
294     public interface Callback {
onMobileDataEnabled(boolean enabled)295         void onMobileDataEnabled(boolean enabled);
296     }
297 }
298