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.messaging.sms;
18 
19 import android.content.ContentValues;
20 import android.content.Context;
21 import android.database.Cursor;
22 import android.database.sqlite.SQLiteDatabase;
23 import android.database.sqlite.SQLiteException;
24 import android.net.Uri;
25 import android.provider.Telephony;
26 import android.support.v7.mms.ApnSettingsLoader;
27 import android.support.v7.mms.MmsManager;
28 import android.text.TextUtils;
29 import android.util.SparseArray;
30 
31 import com.android.messaging.datamodel.data.ParticipantData;
32 import com.android.messaging.mmslib.SqliteWrapper;
33 import com.android.messaging.util.BugleGservices;
34 import com.android.messaging.util.BugleGservicesKeys;
35 import com.android.messaging.util.LogUtil;
36 import com.android.messaging.util.OsUtil;
37 import com.android.messaging.util.PhoneUtils;
38 
39 import java.net.URI;
40 import java.net.URISyntaxException;
41 import java.util.ArrayList;
42 import java.util.List;
43 
44 /**
45  * APN loader for default SMS SIM
46  *
47  * This loader tries to load APNs from 3 sources in order:
48  * 1. Gservices setting
49  * 2. System APN table
50  * 3. Local APN table
51  */
52 public class BugleApnSettingsLoader implements ApnSettingsLoader {
53     /**
54      * The base implementation of an APN
55      */
56     private static class BaseApn implements Apn {
57         /**
58          * Create a base APN from parameters
59          *
60          * @param typesIn the APN type field
61          * @param mmscIn the APN mmsc field
62          * @param proxyIn the APN mmsproxy field
63          * @param portIn the APN mmsport field
64          * @return an instance of base APN, or null if any of the parameter is invalid
65          */
from(final String typesIn, final String mmscIn, final String proxyIn, final String portIn)66         public static BaseApn from(final String typesIn, final String mmscIn, final String proxyIn,
67                 final String portIn) {
68             if (!isValidApnType(trimWithNullCheck(typesIn), APN_TYPE_MMS)) {
69                 return null;
70             }
71             String mmsc = trimWithNullCheck(mmscIn);
72             if (TextUtils.isEmpty(mmsc)) {
73                 return null;
74             }
75             mmsc = trimV4AddrZeros(mmsc);
76             try {
77                 new URI(mmsc);
78             } catch (final URISyntaxException e) {
79                 return null;
80             }
81             String mmsProxy = trimWithNullCheck(proxyIn);
82             int mmsProxyPort = 80;
83             if (!TextUtils.isEmpty(mmsProxy)) {
84                 mmsProxy = trimV4AddrZeros(mmsProxy);
85                 final String portString = trimWithNullCheck(portIn);
86                 if (portString != null) {
87                     try {
88                         mmsProxyPort = Integer.parseInt(portString);
89                     } catch (final NumberFormatException e) {
90                         // Ignore, just use 80 to try
91                     }
92                 }
93             }
94             return new BaseApn(mmsc, mmsProxy, mmsProxyPort);
95         }
96 
97         private final String mMmsc;
98         private final String mMmsProxy;
99         private final int mMmsProxyPort;
100 
BaseApn(final String mmsc, final String proxy, final int port)101         public BaseApn(final String mmsc, final String proxy, final int port) {
102             mMmsc = mmsc;
103             mMmsProxy = proxy;
104             mMmsProxyPort = port;
105         }
106 
107         @Override
getMmsc()108         public String getMmsc() {
109             return mMmsc;
110         }
111 
112         @Override
getMmsProxy()113         public String getMmsProxy() {
114             return mMmsProxy;
115         }
116 
117         @Override
getMmsProxyPort()118         public int getMmsProxyPort() {
119             return mMmsProxyPort;
120         }
121 
122         @Override
setSuccess()123         public void setSuccess() {
124             // Do nothing
125         }
126 
equals(final BaseApn other)127         public boolean equals(final BaseApn other) {
128             return TextUtils.equals(mMmsc, other.getMmsc()) &&
129                     TextUtils.equals(mMmsProxy, other.getMmsProxy()) &&
130                     mMmsProxyPort == other.getMmsProxyPort();
131         }
132     }
133 
134     /**
135      * The APN represented by the local APN table row
136      */
137     private static class DatabaseApn implements Apn {
138         private static final ContentValues CURRENT_NULL_VALUE;
139         private static final ContentValues CURRENT_SET_VALUE;
140         static {
141             CURRENT_NULL_VALUE = new ContentValues(1);
142             CURRENT_NULL_VALUE.putNull(Telephony.Carriers.CURRENT);
143             CURRENT_SET_VALUE = new ContentValues(1);
CURRENT_SET_VALUE.put(Telephony.Carriers.CURRENT, "1")144             CURRENT_SET_VALUE.put(Telephony.Carriers.CURRENT, "1"); // 1 for auto selected APN
145         }
146         private static final String CLEAR_UPDATE_SELECTION = Telephony.Carriers.CURRENT + " =?";
147         private static final String[] CLEAR_UPDATE_SELECTION_ARGS = new String[] { "1" };
148         private static final String SET_UPDATE_SELECTION = Telephony.Carriers._ID + " =?";
149 
150         /**
151          * Create an APN loaded from local database
152          *
153          * @param apns the in-memory APN list
154          * @param typesIn the APN type field
155          * @param mmscIn the APN mmsc field
156          * @param proxyIn the APN mmsproxy field
157          * @param portIn the APN mmsport field
158          * @param rowId the APN's row ID in database
159          * @param current the value of CURRENT column in database
160          * @return an in-memory APN instance for database APN row, null if parameter invalid
161          */
from(final List<Apn> apns, final String typesIn, final String mmscIn, final String proxyIn, final String portIn, final long rowId, final int current)162         public static DatabaseApn from(final List<Apn> apns, final String typesIn,
163                 final String mmscIn, final String proxyIn, final String portIn,
164                 final long rowId, final int current) {
165             if (apns == null) {
166                 return null;
167             }
168             final BaseApn base = BaseApn.from(typesIn, mmscIn, proxyIn, portIn);
169             if (base == null) {
170                 return null;
171             }
172             for (final ApnSettingsLoader.Apn apn : apns) {
173                 if (apn instanceof DatabaseApn && ((DatabaseApn) apn).equals(base)) {
174                     return null;
175                 }
176             }
177             return new DatabaseApn(apns, base, rowId, current);
178         }
179 
180         private final List<Apn> mApns;
181         private final BaseApn mBase;
182         private final long mRowId;
183         private int mCurrent;
184 
DatabaseApn(final List<Apn> apns, final BaseApn base, final long rowId, final int current)185         public DatabaseApn(final List<Apn> apns, final BaseApn base, final long rowId,
186                 final int current) {
187             mApns = apns;
188             mBase = base;
189             mRowId = rowId;
190             mCurrent = current;
191         }
192 
193         @Override
getMmsc()194         public String getMmsc() {
195             return mBase.getMmsc();
196         }
197 
198         @Override
getMmsProxy()199         public String getMmsProxy() {
200             return mBase.getMmsProxy();
201         }
202 
203         @Override
getMmsProxyPort()204         public int getMmsProxyPort() {
205             return mBase.getMmsProxyPort();
206         }
207 
208         @Override
setSuccess()209         public void setSuccess() {
210             moveToListHead();
211             setCurrentInDatabase();
212         }
213 
214         /**
215          * Try to move this APN to the head of in-memory list
216          */
moveToListHead()217         private void moveToListHead() {
218             // If this is being marked as a successful APN, move it to the top of the list so
219             // next time it will be tried first
220             boolean moved = false;
221             synchronized (mApns) {
222                 if (mApns.get(0) != this) {
223                     mApns.remove(this);
224                     mApns.add(0, this);
225                     moved = true;
226                 }
227             }
228             if (moved) {
229                 LogUtil.d(LogUtil.BUGLE_TAG, "Set APN ["
230                         + "MMSC=" + getMmsc() + ", "
231                         + "PROXY=" + getMmsProxy() + ", "
232                         + "PORT=" + getMmsProxyPort() + "] to be first");
233             }
234         }
235 
236         /**
237          * Try to set the APN to be CURRENT in its database table
238          */
setCurrentInDatabase()239         private void setCurrentInDatabase() {
240             synchronized (this) {
241                 if (mCurrent > 0) {
242                     // Already current
243                     return;
244                 }
245                 mCurrent = 1;
246             }
247             LogUtil.d(LogUtil.BUGLE_TAG, "Set APN @" + mRowId + " to be CURRENT in local db");
248             final SQLiteDatabase database = ApnDatabase.getApnDatabase().getWritableDatabase();
249             database.beginTransaction();
250             try {
251                 // clear the previous current=1 apn
252                 // we don't clear current=2 apn since it is manually selected by user
253                 // and we should not override it.
254                 database.update(ApnDatabase.APN_TABLE, CURRENT_NULL_VALUE,
255                         CLEAR_UPDATE_SELECTION, CLEAR_UPDATE_SELECTION_ARGS);
256                 // set this one to be current (1)
257                 database.update(ApnDatabase.APN_TABLE, CURRENT_SET_VALUE, SET_UPDATE_SELECTION,
258                         new String[] { Long.toString(mRowId) });
259                 database.setTransactionSuccessful();
260             } finally {
261                 database.endTransaction();
262             }
263         }
264 
equals(final BaseApn other)265         public boolean equals(final BaseApn other) {
266             if (other == null) {
267                 return false;
268             }
269             return mBase.equals(other);
270         }
271     }
272 
273     /**
274      * APN_TYPE_ALL is a special type to indicate that this APN entry can
275      * service all data connections.
276      */
277     public static final String APN_TYPE_ALL = "*";
278     /** APN type for MMS traffic */
279     public static final String APN_TYPE_MMS = "mms";
280 
281     private static final String[] APN_PROJECTION_SYSTEM = {
282             Telephony.Carriers.TYPE,
283             Telephony.Carriers.MMSC,
284             Telephony.Carriers.MMSPROXY,
285             Telephony.Carriers.MMSPORT,
286     };
287     private static final String[] APN_PROJECTION_LOCAL = {
288             Telephony.Carriers.TYPE,
289             Telephony.Carriers.MMSC,
290             Telephony.Carriers.MMSPROXY,
291             Telephony.Carriers.MMSPORT,
292             Telephony.Carriers.CURRENT,
293             Telephony.Carriers._ID,
294     };
295     private static final int COLUMN_TYPE         = 0;
296     private static final int COLUMN_MMSC         = 1;
297     private static final int COLUMN_MMSPROXY     = 2;
298     private static final int COLUMN_MMSPORT      = 3;
299     private static final int COLUMN_CURRENT      = 4;
300     private static final int COLUMN_ID           = 5;
301 
302     private static final String SELECTION_APN = Telephony.Carriers.APN + "=?";
303     private static final String SELECTION_CURRENT = Telephony.Carriers.CURRENT + " IS NOT NULL";
304     private static final String SELECTION_NUMERIC = Telephony.Carriers.NUMERIC + "=?";
305     private static final String ORDER_BY = Telephony.Carriers.CURRENT + " DESC";
306 
307     private final Context mContext;
308 
309     // Cached APNs for subIds
310     private final SparseArray<List<ApnSettingsLoader.Apn>> mApnsCache;
311 
BugleApnSettingsLoader(final Context context)312     public BugleApnSettingsLoader(final Context context) {
313         mContext = context;
314         mApnsCache = new SparseArray<>();
315     }
316 
317     @Override
get(final String apnName)318     public List<ApnSettingsLoader.Apn> get(final String apnName) {
319         final int subId = PhoneUtils.getDefault().getEffectiveSubId(
320                 ParticipantData.DEFAULT_SELF_SUB_ID);
321         List<ApnSettingsLoader.Apn> apns;
322         boolean didLoad = false;
323         synchronized (this) {
324             apns = mApnsCache.get(subId);
325             if (apns == null) {
326                 apns = new ArrayList<>();
327                 mApnsCache.put(subId, apns);
328                 loadLocked(subId, apnName, apns);
329                 didLoad = true;
330             }
331         }
332         if (didLoad) {
333             LogUtil.i(LogUtil.BUGLE_TAG, "Loaded " + apns.size() + " APNs");
334         }
335         return apns;
336     }
337 
loadLocked(final int subId, final String apnName, final List<Apn> apns)338     private void loadLocked(final int subId, final String apnName, final List<Apn> apns) {
339         // Try Gservices first
340         loadFromGservices(apns);
341         if (apns.size() > 0) {
342             return;
343         }
344         // Try system APN table
345         loadFromSystem(subId, apnName, apns);
346         if (apns.size() > 0) {
347             return;
348         }
349         // Try local APN table
350         loadFromLocalDatabase(apnName, apns);
351         if (apns.size() <= 0) {
352             LogUtil.w(LogUtil.BUGLE_TAG, "Failed to load any APN");
353         }
354     }
355 
356     /**
357      * Load from Gservices if APN setting is set in Gservices
358      *
359      * @param apns the list used to return results
360      */
loadFromGservices(final List<Apn> apns)361     private void loadFromGservices(final List<Apn> apns) {
362         final BugleGservices gservices = BugleGservices.get();
363         final String mmsc = gservices.getString(BugleGservicesKeys.MMS_MMSC, null);
364         if (TextUtils.isEmpty(mmsc)) {
365             return;
366         }
367         LogUtil.i(LogUtil.BUGLE_TAG, "Loading APNs from gservices");
368         final String proxy = gservices.getString(BugleGservicesKeys.MMS_PROXY_ADDRESS, null);
369         final int port = gservices.getInt(BugleGservicesKeys.MMS_PROXY_PORT, -1);
370         final Apn apn = BaseApn.from("mms", mmsc, proxy, Integer.toString(port));
371         if (apn != null) {
372             apns.add(apn);
373         }
374     }
375 
376     /**
377      * Load matching APNs from telephony provider.
378      * We try different combinations of the query to work around some platform quirks.
379      *
380      * @param subId the SIM subId
381      * @param apnName the APN name to match
382      * @param apns the list used to return results
383      */
loadFromSystem(final int subId, final String apnName, final List<Apn> apns)384     private void loadFromSystem(final int subId, final String apnName, final List<Apn> apns) {
385         Uri uri;
386         if (OsUtil.isAtLeastL_MR1() && subId != MmsManager.DEFAULT_SUB_ID) {
387             uri = Uri.withAppendedPath(Telephony.Carriers.CONTENT_URI, "/subId/" + subId);
388         } else {
389             uri = Telephony.Carriers.CONTENT_URI;
390         }
391         Cursor cursor = null;
392         try {
393             for (; ; ) {
394                 // Try different combinations of queries. Some would work on some platforms.
395                 // So we query each combination until we find one returns non-empty result.
396                 cursor = querySystem(uri, true/*checkCurrent*/, apnName);
397                 if (cursor != null) {
398                     break;
399                 }
400                 cursor = querySystem(uri, false/*checkCurrent*/, apnName);
401                 if (cursor != null) {
402                     break;
403                 }
404                 cursor = querySystem(uri, true/*checkCurrent*/, null/*apnName*/);
405                 if (cursor != null) {
406                     break;
407                 }
408                 cursor = querySystem(uri, false/*checkCurrent*/, null/*apnName*/);
409                 break;
410             }
411         } catch (final SecurityException e) {
412             // Can't access platform APN table, return directly
413             return;
414         }
415         if (cursor == null) {
416             return;
417         }
418         try {
419             if (cursor.moveToFirst()) {
420                 final ApnSettingsLoader.Apn apn = BaseApn.from(
421                         cursor.getString(COLUMN_TYPE),
422                         cursor.getString(COLUMN_MMSC),
423                         cursor.getString(COLUMN_MMSPROXY),
424                         cursor.getString(COLUMN_MMSPORT));
425                 if (apn != null) {
426                     apns.add(apn);
427                 }
428             }
429         } finally {
430             cursor.close();
431         }
432     }
433 
434     /**
435      * Query system APN table
436      *
437      * @param uri The APN query URL to use
438      * @param checkCurrent If add "CURRENT IS NOT NULL" condition
439      * @param apnName The optional APN name for query condition
440      * @return A cursor of the query result. If a cursor is returned as not null, it is
441      *         guaranteed to contain at least one row.
442      */
querySystem(final Uri uri, final boolean checkCurrent, String apnName)443     private Cursor querySystem(final Uri uri, final boolean checkCurrent, String apnName) {
444         LogUtil.i(LogUtil.BUGLE_TAG, "Loading APNs from system, "
445                 + "checkCurrent=" + checkCurrent + " apnName=" + apnName);
446         final StringBuilder selectionBuilder = new StringBuilder();
447         String[] selectionArgs = null;
448         if (checkCurrent) {
449             selectionBuilder.append(SELECTION_CURRENT);
450         }
451         apnName = trimWithNullCheck(apnName);
452         if (!TextUtils.isEmpty(apnName)) {
453             if (selectionBuilder.length() > 0) {
454                 selectionBuilder.append(" AND ");
455             }
456             selectionBuilder.append(SELECTION_APN);
457             selectionArgs = new String[] { apnName };
458         }
459         try {
460             final Cursor cursor = SqliteWrapper.query(
461                     mContext,
462                     mContext.getContentResolver(),
463                     uri,
464                     APN_PROJECTION_SYSTEM,
465                     selectionBuilder.toString(),
466                     selectionArgs,
467                     null/*sortOrder*/);
468             if (cursor == null || cursor.getCount() < 1) {
469                 if (cursor != null) {
470                     cursor.close();
471                 }
472                 LogUtil.w(LogUtil.BUGLE_TAG, "Query " + uri + " with apn " + apnName + " and "
473                         + (checkCurrent ? "checking CURRENT" : "not checking CURRENT")
474                         + " returned empty");
475                 return null;
476             }
477             return cursor;
478         } catch (final SQLiteException e) {
479             LogUtil.w(LogUtil.BUGLE_TAG, "APN table query exception: " + e);
480         } catch (final SecurityException e) {
481             LogUtil.w(LogUtil.BUGLE_TAG, "Platform restricts APN table access: " + e);
482             throw e;
483         }
484         return null;
485     }
486 
487     /**
488      * Load matching APNs from local APN table.
489      * We try both using the APN name and not using the APN name.
490      *
491      * @param apnName the APN name
492      * @param apns the list of results to return
493      */
loadFromLocalDatabase(final String apnName, final List<Apn> apns)494     private void loadFromLocalDatabase(final String apnName, final List<Apn> apns) {
495         LogUtil.i(LogUtil.BUGLE_TAG, "Loading APNs from local APN table");
496         final SQLiteDatabase database = ApnDatabase.getApnDatabase().getWritableDatabase();
497         final String mccMnc = PhoneUtils.getMccMncString(PhoneUtils.getDefault().getMccMnc());
498         Cursor cursor = null;
499         cursor = queryLocalDatabase(database, mccMnc, apnName);
500         if (cursor == null) {
501             cursor = queryLocalDatabase(database, mccMnc, null/*apnName*/);
502         }
503         if (cursor == null) {
504             LogUtil.w(LogUtil.BUGLE_TAG, "Could not find any APN in local table");
505             return;
506         }
507         try {
508             while (cursor.moveToNext()) {
509                 final Apn apn = DatabaseApn.from(apns,
510                         cursor.getString(COLUMN_TYPE),
511                         cursor.getString(COLUMN_MMSC),
512                         cursor.getString(COLUMN_MMSPROXY),
513                         cursor.getString(COLUMN_MMSPORT),
514                         cursor.getLong(COLUMN_ID),
515                         cursor.getInt(COLUMN_CURRENT));
516                 if (apn != null) {
517                     apns.add(apn);
518                 }
519             }
520         } finally {
521             cursor.close();
522         }
523     }
524 
525     /**
526      * Make a query of local APN table based on MCC/MNC and APN name, sorted by CURRENT
527      * column in descending order
528      *
529      * @param db the local database
530      * @param numeric the MCC/MNC string
531      * @param apnName the optional APN name to match
532      * @return the cursor of the query, null if no result
533      */
queryLocalDatabase(final SQLiteDatabase db, final String numeric, final String apnName)534     private static Cursor queryLocalDatabase(final SQLiteDatabase db, final String numeric,
535             final String apnName) {
536         final String selection;
537         final String[] selectionArgs;
538         if (TextUtils.isEmpty(apnName)) {
539             selection = SELECTION_NUMERIC;
540             selectionArgs = new String[] { numeric };
541         } else {
542             selection = SELECTION_NUMERIC + " AND " + SELECTION_APN;
543             selectionArgs = new String[] { numeric, apnName };
544         }
545         Cursor cursor = null;
546         try {
547             cursor = db.query(ApnDatabase.APN_TABLE, APN_PROJECTION_LOCAL, selection, selectionArgs,
548                     null/*groupBy*/, null/*having*/, ORDER_BY, null/*limit*/);
549         } catch (final SQLiteException e) {
550             LogUtil.w(LogUtil.BUGLE_TAG, "Local APN table does not exist. Try rebuilding.", e);
551             ApnDatabase.forceBuildAndLoadApnTables();
552             cursor = db.query(ApnDatabase.APN_TABLE, APN_PROJECTION_LOCAL, selection, selectionArgs,
553                     null/*groupBy*/, null/*having*/, ORDER_BY, null/*limit*/);
554         }
555         if (cursor == null || cursor.getCount() < 1) {
556             if (cursor != null) {
557                 cursor.close();
558             }
559             LogUtil.w(LogUtil.BUGLE_TAG, "Query local APNs with apn " + apnName
560                     + " returned empty");
561             return null;
562         }
563         return cursor;
564     }
565 
trimWithNullCheck(final String value)566     private static String trimWithNullCheck(final String value) {
567         return value != null ? value.trim() : null;
568     }
569 
570     /**
571      * Trim leading zeros from IPv4 address strings
572      * Our base libraries will interpret that as octel..
573      * Must leave non v4 addresses and host names alone.
574      * For example, 192.168.000.010 -> 192.168.0.10
575      *
576      * @param addr a string representing an ip addr
577      * @return a string propertly trimmed
578      */
trimV4AddrZeros(final String addr)579     private static String trimV4AddrZeros(final String addr) {
580         if (addr == null) {
581             return null;
582         }
583         final String[] octets = addr.split("\\.");
584         if (octets.length != 4) {
585             return addr;
586         }
587         final StringBuilder builder = new StringBuilder(16);
588         String result = null;
589         for (int i = 0; i < 4; i++) {
590             try {
591                 if (octets[i].length() > 3) {
592                     return addr;
593                 }
594                 builder.append(Integer.parseInt(octets[i]));
595             } catch (final NumberFormatException e) {
596                 return addr;
597             }
598             if (i < 3) {
599                 builder.append('.');
600             }
601         }
602         result = builder.toString();
603         return result;
604     }
605 
606     /**
607      * Check if the APN contains the APN type we want
608      *
609      * @param types The string encodes a list of supported types
610      * @param requestType The type we want
611      * @return true if the input types string contains the requestType
612      */
isValidApnType(final String types, final String requestType)613     public static boolean isValidApnType(final String types, final String requestType) {
614         // If APN type is unspecified, assume APN_TYPE_ALL.
615         if (TextUtils.isEmpty(types)) {
616             return true;
617         }
618         for (final String t : types.split(",")) {
619             if (t.equals(requestType) || t.equals(APN_TYPE_ALL)) {
620                 return true;
621             }
622         }
623         return false;
624     }
625 
626     /**
627      * Get the ID of first APN to try
628      */
getFirstTryApn(final SQLiteDatabase database, final String mccMnc)629     public static String getFirstTryApn(final SQLiteDatabase database, final String mccMnc) {
630         String key = null;
631         Cursor cursor = null;
632         try {
633             cursor = queryLocalDatabase(database, mccMnc, null/*apnName*/);
634             if (cursor.moveToFirst()) {
635                 key = cursor.getString(ApnDatabase.COLUMN_ID);
636             }
637         } catch (final Exception e) {
638             // Nothing to do
639         } finally {
640             if (cursor != null) {
641                 cursor.close();
642             }
643         }
644         return key;
645     }
646 }
647