| /* |
| * Copyright (C) 2022 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package com.android.phone.slice; |
| |
| import static android.telephony.TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_ALREADY_IN_PROGRESS; |
| import static android.telephony.TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_ALREADY_PURCHASED; |
| import static android.telephony.TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_CARRIER_DISABLED; |
| import static android.telephony.TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_ENTITLEMENT_CHECK_FAILED; |
| import static android.telephony.TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_NOT_DEFAULT_DATA_SUBSCRIPTION; |
| |
| import android.annotation.IntDef; |
| import android.annotation.NonNull; |
| import android.annotation.Nullable; |
| import android.app.PendingIntent; |
| import android.content.BroadcastReceiver; |
| import android.content.ComponentName; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.content.IntentFilter; |
| import android.content.SharedPreferences; |
| import android.net.ConnectivityManager; |
| import android.os.AsyncResult; |
| import android.os.Handler; |
| import android.os.HandlerThread; |
| import android.os.Looper; |
| import android.os.Message; |
| import android.os.PersistableBundle; |
| import android.provider.DeviceConfig; |
| import android.sysprop.TelephonyProperties; |
| import android.telephony.AnomalyReporter; |
| import android.telephony.CarrierConfigManager; |
| import android.telephony.SubscriptionManager; |
| import android.telephony.TelephonyManager; |
| import android.telephony.data.NetworkSliceInfo; |
| import android.telephony.data.NetworkSlicingConfig; |
| import android.text.TextUtils; |
| import android.util.Log; |
| import android.webkit.URLUtil; |
| import android.webkit.WebView; |
| |
| import com.android.internal.annotations.VisibleForTesting; |
| import com.android.internal.telephony.Phone; |
| |
| import java.lang.annotation.Retention; |
| import java.lang.annotation.RetentionPolicy; |
| import java.net.MalformedURLException; |
| import java.net.URISyntaxException; |
| import java.net.URL; |
| import java.time.LocalDate; |
| import java.time.ZoneId; |
| import java.time.format.DateTimeParseException; |
| import java.util.Arrays; |
| import java.util.HashMap; |
| import java.util.HashSet; |
| import java.util.Map; |
| import java.util.Set; |
| import java.util.UUID; |
| import java.util.concurrent.Executor; |
| import java.util.concurrent.TimeUnit; |
| import java.util.function.Consumer; |
| |
| /** |
| * The SlicePurchaseController controls the purchase and availability of all cellular premium |
| * capabilities. Applications can check whether premium capabilities are available by calling |
| * {@link TelephonyManager#isPremiumCapabilityAvailableForPurchase(int)}. If this returns true, |
| * they can then call {@link TelephonyManager#purchasePremiumCapability(int, Executor, Consumer)} |
| * to purchase the premium capability. If all conditions are met, a notification will be displayed |
| * to the user prompting them to purchase the premium capability. If the user confirms on the |
| * notification, a {@link WebView} will open that allows the user to purchase the premium capability |
| * from the carrier. If the purchase is successful, the premium capability will be available for |
| * all applications to request through {@link ConnectivityManager#requestNetwork}. |
| */ |
| public class SlicePurchaseController extends Handler { |
| @NonNull private static final String TAG = "SlicePurchaseController"; |
| |
| /** Unknown failure code. */ |
| public static final int FAILURE_CODE_UNKNOWN = 0; |
| /** Performance boost purchase failed because the carrier URL is unavailable. */ |
| public static final int FAILURE_CODE_CARRIER_URL_UNAVAILABLE = 1; |
| /** Performance boost purchase failed because the server is unreachable. */ |
| public static final int FAILURE_CODE_SERVER_UNREACHABLE = 2; |
| /** Performance boost purchase failed because user authentication failed. */ |
| public static final int FAILURE_CODE_AUTHENTICATION_FAILED = 3; |
| /** Performance boost purchase failed because the payment failed. */ |
| public static final int FAILURE_CODE_PAYMENT_FAILED = 4; |
| |
| /** |
| * Failure codes that the carrier website can return when a premium capability purchase fails. |
| */ |
| @Retention(RetentionPolicy.SOURCE) |
| @IntDef(prefix = { "FAILURE_CODE_" }, value = { |
| FAILURE_CODE_UNKNOWN, |
| FAILURE_CODE_CARRIER_URL_UNAVAILABLE, |
| FAILURE_CODE_SERVER_UNREACHABLE, |
| FAILURE_CODE_AUTHENTICATION_FAILED, |
| FAILURE_CODE_PAYMENT_FAILED}) |
| public @interface FailureCode {} |
| |
| /** Value for an invalid premium capability. */ |
| public static final int PREMIUM_CAPABILITY_INVALID = -1; |
| |
| /** Asset URL for the slice_purchase_test.html file. */ |
| public static final String SLICE_PURCHASE_TEST_FILE = |
| "file:///android_asset/slice_purchase_test.html"; |
| |
| /** Purchasing the premium capability is no longer throttled. */ |
| private static final int EVENT_PURCHASE_UNTHROTTLED = 1; |
| /** Slicing config changed. */ |
| private static final int EVENT_SLICING_CONFIG_CHANGED = 2; |
| /** Start slice purchase application. */ |
| private static final int EVENT_START_SLICE_PURCHASE_APP = 3; |
| /** |
| * Premium capability was not purchased within the timeout specified by |
| * {@link CarrierConfigManager#KEY_PREMIUM_CAPABILITY_NOTIFICATION_DISPLAY_TIMEOUT_MILLIS_LONG}. |
| */ |
| private static final int EVENT_PURCHASE_TIMEOUT = 4; |
| /** |
| * Network did not set up the slicing configuration within the timeout specified by |
| * {@link CarrierConfigManager#KEY_PREMIUM_CAPABILITY_NETWORK_SETUP_TIME_MILLIS_LONG}. |
| */ |
| private static final int EVENT_SETUP_TIMEOUT = 5; |
| /** Device config changed. */ |
| private static final int EVENT_DEVICE_CONFIG_CHANGED = 6; |
| |
| /** UUID to report an anomaly when a premium capability is throttled twice in a row. */ |
| private static final String UUID_CAPABILITY_THROTTLED_TWICE = |
| "15574927-e2e2-4593-99d4-2f340d22b383"; |
| /** UUID to report an anomaly when receiving an invalid phone ID. */ |
| private static final String UUID_INVALID_PHONE_ID = "ced79f1a-8ac0-4260-8cf3-08b54c0494f3"; |
| /** UUID to report an anomaly when receiving an unknown action. */ |
| private static final String UUID_UNKNOWN_ACTION = "0197efb0-dab1-4b0a-abaf-ac9336ec7923"; |
| /** UUID to report an anomaly when receiving an unknown failure code with a non-empty reason. */ |
| private static final String UUID_UNKNOWN_FAILURE_CODE = "76943b23-4415-400c-9855-b534fc4fc62c"; |
| /** |
| * UUID to report an anomaly when the network fails to set up a slicing configuration after |
| * the user purchases a premium capability. |
| */ |
| private static final String UUID_NETWORK_SETUP_FAILED = "12eeffbf-08f8-40ed-9a00-d344199552fc"; |
| |
| /** |
| * Action to start the slice purchase application and display the |
| * performance boost notification. |
| */ |
| public static final String ACTION_START_SLICE_PURCHASE_APP = |
| "com.android.phone.slice.action.START_SLICE_PURCHASE_APP"; |
| /** Action indicating the premium capability purchase was not completed in time. */ |
| public static final String ACTION_SLICE_PURCHASE_APP_RESPONSE_TIMEOUT = |
| "com.android.phone.slice.action.SLICE_PURCHASE_APP_RESPONSE_TIMEOUT"; |
| /** Action indicating the performance boost notification or WebView was canceled. */ |
| private static final String ACTION_SLICE_PURCHASE_APP_RESPONSE_CANCELED = |
| "com.android.phone.slice.action.SLICE_PURCHASE_APP_RESPONSE_CANCELED"; |
| /** Action indicating a carrier error prevented premium capability purchase. */ |
| private static final String ACTION_SLICE_PURCHASE_APP_RESPONSE_CARRIER_ERROR = |
| "com.android.phone.slice.action.SLICE_PURCHASE_APP_RESPONSE_CARRIER_ERROR"; |
| /** |
| * Action indicating a Telephony or slice purchase application error prevented premium |
| * capability purchase. |
| */ |
| private static final String ACTION_SLICE_PURCHASE_APP_RESPONSE_REQUEST_FAILED = |
| "com.android.phone.slice.action.SLICE_PURCHASE_APP_RESPONSE_REQUEST_FAILED"; |
| /** Action indicating the purchase request was not made on the default data subscription. */ |
| private static final String ACTION_SLICE_PURCHASE_APP_RESPONSE_NOT_DEFAULT_DATA_SUBSCRIPTION = |
| "com.android.phone.slice.action." |
| + "SLICE_PURCHASE_APP_RESPONSE_NOT_DEFAULT_DATA_SUBSCRIPTION"; |
| /** Action indicating the purchase request was successful. */ |
| private static final String ACTION_SLICE_PURCHASE_APP_RESPONSE_SUCCESS = |
| "com.android.phone.slice.action.SLICE_PURCHASE_APP_RESPONSE_SUCCESS"; |
| /** |
| * Action indicating the slice purchase application showed the performance boost notification. |
| */ |
| private static final String ACTION_SLICE_PURCHASE_APP_RESPONSE_NOTIFICATION_SHOWN = |
| "com.android.phone.slice.action.SLICE_PURCHASE_APP_RESPONSE_NOTIFICATION_SHOWN"; |
| |
| /** Extra for the phone index to send to the slice purchase application. */ |
| public static final String EXTRA_PHONE_ID = "com.android.phone.slice.extra.PHONE_ID"; |
| /** Extra for the subscription ID to send to the slice purchase application. */ |
| public static final String EXTRA_SUB_ID = "com.android.phone.slice.extra.SUB_ID"; |
| /** |
| * Extra for the requested premium capability to purchase from the slice purchase application. |
| */ |
| public static final String EXTRA_PREMIUM_CAPABILITY = |
| "com.android.phone.slice.extra.PREMIUM_CAPABILITY"; |
| /** Extra for the carrier URL to display to the user to allow premium capability purchase. */ |
| public static final String EXTRA_PURCHASE_URL = "com.android.phone.slice.extra.PURCHASE_URL"; |
| /** Extra for the duration of the purchased premium capability. */ |
| public static final String EXTRA_PURCHASE_DURATION = |
| "com.android.phone.slice.extra.PURCHASE_DURATION"; |
| /** Extra for the {@link FailureCode} why the premium capability purchase failed. */ |
| public static final String EXTRA_FAILURE_CODE = "com.android.phone.slice.extra.FAILURE_CODE"; |
| /** Extra for the human-readable reason why the premium capability purchase failed. */ |
| public static final String EXTRA_FAILURE_REASON = |
| "com.android.phone.slice.extra.FAILURE_REASON"; |
| /** |
| * Extra for the user's carrier. |
| */ |
| public static final String EXTRA_CARRIER = "com.android.phone.slice.extra.CARRIER"; |
| /** |
| * Extra for the canceled PendingIntent that the slice purchase application can send as a |
| * response if the performance boost notification or WebView was canceled by the user. |
| * Sends {@link #ACTION_SLICE_PURCHASE_APP_RESPONSE_CANCELED}. |
| */ |
| public static final String EXTRA_INTENT_CANCELED = |
| "com.android.phone.slice.extra.INTENT_CANCELED"; |
| /** |
| * Extra for the carrier error PendingIntent that the slice purchase application can send as a |
| * response if the premium capability purchase request failed due to a carrier error. |
| * Sends {@link #ACTION_SLICE_PURCHASE_APP_RESPONSE_CARRIER_ERROR}. |
| * Sender can modify the intent to specify the failure code and reason for failure with |
| * {@link #EXTRA_FAILURE_CODE} and {@link #EXTRA_FAILURE_REASON}. |
| */ |
| public static final String EXTRA_INTENT_CARRIER_ERROR = |
| "com.android.phone.slice.extra.INTENT_CARRIER_ERROR"; |
| /** |
| * Extra for the request failed PendingIntent that the slice purchase application can send as a |
| * response if the premium capability purchase request failed due to an error in Telephony or |
| * the slice purchase application. |
| * Sends {@link #ACTION_SLICE_PURCHASE_APP_RESPONSE_REQUEST_FAILED}. |
| */ |
| public static final String EXTRA_INTENT_REQUEST_FAILED = |
| "com.android.phone.slice.extra.INTENT_REQUEST_FAILED"; |
| /** |
| * Extra for the not-default data subscription ID PendingIntent that the slice purchase |
| * application can send as a response if the premium capability purchase request failed because |
| * it was not requested on the default data subscription. |
| * Sends {@link #ACTION_SLICE_PURCHASE_APP_RESPONSE_NOT_DEFAULT_DATA_SUBSCRIPTION}. |
| */ |
| public static final String EXTRA_INTENT_NOT_DEFAULT_DATA_SUBSCRIPTION = |
| "com.android.phone.slice.extra.INTENT_NOT_DEFAULT_DATA_SUBSCRIPTION"; |
| /** |
| * Extra for the success PendingIntent that the slice purchase application can send as a |
| * response if the premium capability purchase request was successful. |
| * Sends {@link #ACTION_SLICE_PURCHASE_APP_RESPONSE_SUCCESS}. |
| * Sender can modify the intent to specify a purchase duration with |
| * {@link #EXTRA_PURCHASE_DURATION}. |
| */ |
| public static final String EXTRA_INTENT_SUCCESS = |
| "com.android.phone.slice.extra.INTENT_SUCCESS"; |
| /** |
| * Extra for the PendingIntent that the slice purchase application can send to indicate |
| * that it displayed the performance boost notification to the user. |
| * Sends {@link #ACTION_SLICE_PURCHASE_APP_RESPONSE_NOTIFICATION_SHOWN}. |
| */ |
| public static final String EXTRA_INTENT_NOTIFICATION_SHOWN = |
| "com.android.phone.slice.extra.NOTIFICATION_SHOWN"; |
| |
| /** Component name for the SlicePurchaseBroadcastReceiver. */ |
| private static final ComponentName SLICE_PURCHASE_APP_COMPONENT_NAME = |
| ComponentName.unflattenFromString( |
| "com.android.carrierdefaultapp/.SlicePurchaseBroadcastReceiver"); |
| |
| /** Shared preference name for performance boost notification preferences. */ |
| private static final String PERFORMANCE_BOOST_NOTIFICATION_PREFERENCES = |
| "performance_boost_notification_preferences"; |
| /** Shared preference key for daily count of performance boost notifications. */ |
| private static final String KEY_DAILY_NOTIFICATION_COUNT = "daily_notification_count"; |
| /** Shared preference key for monthly count of performance boost notifications. */ |
| private static final String KEY_MONTHLY_NOTIFICATION_COUNT = "monthly_notification_count"; |
| /** DeviceConfig key for whether the slicing upsell feature is enabled. */ |
| private static final String KEY_ENABLE_SLICING_UPSELL = "enable_slicing_upsell"; |
| /** |
| * Shared preference key for the date the daily or monthly counts of performance boost |
| * notifications were last reset. |
| * A String with ISO-8601 format {@code YYYY-MM-DD}, from {@link LocalDate#toString}. |
| * For example, if the count was last updated on December 25, 2020, this would be `2020-12-25`. |
| */ |
| private static final String KEY_NOTIFICATION_COUNT_LAST_RESET_DATE = |
| "notification_count_last_reset_date"; |
| |
| /** Map of phone ID -> SlicePurchaseController instances. */ |
| @NonNull private static final Map<Integer, SlicePurchaseController> sInstances = |
| new HashMap<>(); |
| |
| /** The Phone instance used to create the SlicePurchaseController. */ |
| @NonNull private final Phone mPhone; |
| /** The set of capabilities that are pending network setup. */ |
| @NonNull private final Set<Integer> mPendingSetupCapabilities = new HashSet<>(); |
| /** The set of throttled capabilities. */ |
| @NonNull private final Set<Integer> mThrottledCapabilities = new HashSet<>(); |
| /** A map of pending capabilities to the onComplete message for the purchase request. */ |
| @NonNull private final Map<Integer, Message> mPendingPurchaseCapabilities = new HashMap<>(); |
| /** |
| * A map of capabilities to the SlicePurchaseControllerBroadcastReceiver to handle |
| * slice purchase application responses. |
| */ |
| @NonNull private final Map<Integer, SlicePurchaseControllerBroadcastReceiver> |
| mSlicePurchaseControllerBroadcastReceivers = new HashMap<>(); |
| /** The current network slicing configuration. */ |
| @Nullable private NetworkSlicingConfig mSlicingConfig; |
| |
| /** LocalDate to use when resetting notification counts. {@code null} except when testing. */ |
| @Nullable private LocalDate mLocalDate; |
| /** The number of times the performance boost notification has been shown today. */ |
| private int mDailyCount; |
| /** The number of times the performance boost notification has been shown this month. */ |
| private int mMonthlyCount; |
| /** {@code true} if the slicing upsell feature is enabled and {@code false} otherwise. */ |
| private boolean mIsSlicingUpsellEnabled; |
| |
| /** |
| * BroadcastReceiver to receive responses from the slice purchase application. |
| */ |
| private class SlicePurchaseControllerBroadcastReceiver extends BroadcastReceiver { |
| @TelephonyManager.PremiumCapability private final int mCapability; |
| |
| /** |
| * Create a SlicePurchaseControllerBroadcastReceiver for the given capability |
| * |
| * @param capability The requested capability to listen to response for. |
| */ |
| SlicePurchaseControllerBroadcastReceiver( |
| @TelephonyManager.PremiumCapability int capability) { |
| mCapability = capability; |
| } |
| |
| /** |
| * Process responses from the slice purchase application. |
| * |
| * @param context The Context in which the receiver is running. |
| * @param intent The Intent being received. |
| */ |
| @Override |
| public void onReceive(@NonNull Context context, @NonNull Intent intent) { |
| String action = intent.getAction(); |
| logd("SlicePurchaseControllerBroadcastReceiver(" |
| + TelephonyManager.convertPremiumCapabilityToString(mCapability) |
| + ") received action: " + action); |
| int phoneId = intent.getIntExtra(EXTRA_PHONE_ID, |
| SubscriptionManager.INVALID_PHONE_INDEX); |
| int capability = intent.getIntExtra(EXTRA_PREMIUM_CAPABILITY, |
| PREMIUM_CAPABILITY_INVALID); |
| if (SlicePurchaseController.getInstance(phoneId) == null) { |
| reportAnomaly(UUID_INVALID_PHONE_ID, "SlicePurchaseControllerBroadcastReceiver( " |
| + TelephonyManager.convertPremiumCapabilityToString(mCapability) |
| + ") received invalid phoneId: " + phoneId); |
| return; |
| } else if (capability != mCapability) { |
| logd("SlicePurchaseControllerBroadcastReceiver(" |
| + TelephonyManager.convertPremiumCapabilityToString(mCapability) |
| + ") ignoring intent for capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| return; |
| } |
| switch (action) { |
| case ACTION_SLICE_PURCHASE_APP_RESPONSE_CANCELED: { |
| logd("Slice purchase application canceled for capability: " |
| + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| SlicePurchaseController.getInstance(phoneId) |
| .handlePurchaseResult(capability, |
| TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_USER_CANCELED, |
| true); |
| break; |
| } |
| case ACTION_SLICE_PURCHASE_APP_RESPONSE_CARRIER_ERROR: { |
| int failureCode = intent.getIntExtra(EXTRA_FAILURE_CODE, FAILURE_CODE_UNKNOWN); |
| String failureReason = intent.getStringExtra(EXTRA_FAILURE_REASON); |
| SlicePurchaseController.getInstance(phoneId).onCarrierError( |
| capability, failureCode, failureReason); |
| break; |
| } |
| case ACTION_SLICE_PURCHASE_APP_RESPONSE_REQUEST_FAILED: { |
| logd("Purchase premium capability request failed for capability: " |
| + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| SlicePurchaseController.getInstance(phoneId) |
| .handlePurchaseResult(capability, |
| TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_REQUEST_FAILED, |
| false); |
| break; |
| } |
| case ACTION_SLICE_PURCHASE_APP_RESPONSE_NOT_DEFAULT_DATA_SUBSCRIPTION: { |
| logd("Purchase premium capability request was not made on the default data " |
| + "subscription for capability: " |
| + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| SlicePurchaseController.getInstance(phoneId) |
| .handlePurchaseResult(capability, |
| PURCHASE_PREMIUM_CAPABILITY_RESULT_NOT_DEFAULT_DATA_SUBSCRIPTION, |
| false); |
| break; |
| } |
| case ACTION_SLICE_PURCHASE_APP_RESPONSE_SUCCESS: { |
| long duration = intent.getLongExtra(EXTRA_PURCHASE_DURATION, 0); |
| SlicePurchaseController.getInstance(phoneId).onCarrierSuccess( |
| capability, duration); |
| break; |
| } |
| case ACTION_SLICE_PURCHASE_APP_RESPONSE_NOTIFICATION_SHOWN: { |
| SlicePurchaseController.getInstance(phoneId).onNotificationShown(); |
| break; |
| } |
| default: |
| reportAnomaly(UUID_UNKNOWN_ACTION, "SlicePurchaseControllerBroadcastReceiver(" |
| + TelephonyManager.convertPremiumCapabilityToString(mCapability) |
| + ") received unknown action: " + action); |
| break; |
| } |
| } |
| } |
| |
| /** |
| * Get the static SlicePurchaseController instance for the given phone or create one if it |
| * doesn't exist. |
| * |
| * @param phone The Phone to get the SlicePurchaseController for. |
| * @return The static SlicePurchaseController instance. |
| */ |
| @NonNull public static synchronized SlicePurchaseController getInstance(@NonNull Phone phone) { |
| // TODO: Add listeners for multi sim setting changed (maybe carrier config changed too) |
| // that dismiss notifications and update SlicePurchaseController instance |
| int phoneId = phone.getPhoneId(); |
| if (sInstances.get(phoneId) == null) { |
| HandlerThread handlerThread = new HandlerThread("SlicePurchaseController"); |
| handlerThread.start(); |
| sInstances.put(phoneId, new SlicePurchaseController(phone, handlerThread.getLooper())); |
| } |
| return sInstances.get(phoneId); |
| } |
| |
| /** |
| * Get the static SlicePurchaseController instance for the given phone ID if it exists. |
| * |
| * @param phoneId The phone ID to get the SlicePurchaseController for. |
| * @return The static SlicePurchaseController instance or |
| * {@code null} if it hasn't been created yet. |
| */ |
| @Nullable private static SlicePurchaseController getInstance(int phoneId) { |
| return sInstances.get(phoneId); |
| } |
| |
| /** |
| * Create a SlicePurchaseController for the given phone on the given looper. |
| * |
| * @param phone The Phone to create the SlicePurchaseController for. |
| * @param looper The Looper to run the SlicePurchaseController on. |
| */ |
| @VisibleForTesting |
| public SlicePurchaseController(@NonNull Phone phone, @NonNull Looper looper) { |
| super(looper); |
| mPhone = phone; |
| // TODO: Create a cached value for slicing config in DataIndication and initialize here |
| mPhone.mCi.registerForSlicingConfigChanged(this, EVENT_SLICING_CONFIG_CHANGED, null); |
| mIsSlicingUpsellEnabled = DeviceConfig.getBoolean( |
| DeviceConfig.NAMESPACE_TELEPHONY, KEY_ENABLE_SLICING_UPSELL, false); |
| DeviceConfig.addOnPropertiesChangedListener( |
| DeviceConfig.NAMESPACE_TELEPHONY, this::post, |
| properties -> { |
| if (TextUtils.equals(DeviceConfig.NAMESPACE_TELEPHONY, |
| properties.getNamespace())) { |
| sendEmptyMessage(EVENT_DEVICE_CONFIG_CHANGED); |
| } |
| }); |
| updateNotificationCounts(); |
| } |
| |
| /** |
| * Set the LocalDate to use for resetting daily and monthly notification counts. |
| * |
| * @param localDate The LocalDate instance to use. |
| */ |
| @VisibleForTesting |
| public void setLocalDate(@NonNull LocalDate localDate) { |
| mLocalDate = localDate; |
| } |
| |
| @Override |
| public void handleMessage(@NonNull Message msg) { |
| switch (msg.what) { |
| case EVENT_PURCHASE_UNTHROTTLED: { |
| int capability = (int) msg.obj; |
| logd("EVENT_PURCHASE_UNTHROTTLED: for capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| mThrottledCapabilities.remove(capability); |
| break; |
| } |
| case EVENT_SLICING_CONFIG_CHANGED: { |
| AsyncResult ar = (AsyncResult) msg.obj; |
| NetworkSlicingConfig config = (NetworkSlicingConfig) ar.result; |
| logd("EVENT_SLICING_CONFIG_CHANGED: from " + mSlicingConfig + " to " + config); |
| mSlicingConfig = config; |
| onSlicingConfigChanged(); |
| break; |
| } |
| case EVENT_START_SLICE_PURCHASE_APP: { |
| int capability = (int) msg.obj; |
| logd("EVENT_START_SLICE_PURCHASE_APP: " |
| + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| onStartSlicePurchaseApplication(capability); |
| break; |
| } |
| case EVENT_PURCHASE_TIMEOUT: { |
| int capability = (int) msg.obj; |
| logd("EVENT_PURCHASE_TIMEOUT: for capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| onTimeout(capability); |
| break; |
| } |
| case EVENT_SETUP_TIMEOUT: |
| int capability = (int) msg.obj; |
| logd("EVENT_SETUP_TIMEOUT: for capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| onSetupTimeout(capability); |
| break; |
| case EVENT_DEVICE_CONFIG_CHANGED: |
| boolean isSlicingUpsellEnabled = DeviceConfig.getBoolean( |
| DeviceConfig.NAMESPACE_TELEPHONY, KEY_ENABLE_SLICING_UPSELL, false); |
| if (isSlicingUpsellEnabled != mIsSlicingUpsellEnabled) { |
| logd("EVENT_DEVICE_CONFIG_CHANGED: from " + mIsSlicingUpsellEnabled + " to " |
| + isSlicingUpsellEnabled); |
| mIsSlicingUpsellEnabled = isSlicingUpsellEnabled; |
| } |
| break; |
| default: |
| loge("Unknown event: " + msg.obj); |
| } |
| } |
| |
| /** |
| * Check whether the given premium capability is available for purchase from the carrier. |
| * |
| * @param capability The premium capability to check. |
| * @return Whether the given premium capability is available to purchase. |
| */ |
| public boolean isPremiumCapabilityAvailableForPurchase( |
| @TelephonyManager.PremiumCapability int capability) { |
| if (!arePremiumCapabilitiesSupportedByDevice()) { |
| logd("Premium capabilities unsupported by the device."); |
| return false; |
| } |
| if (!isPremiumCapabilitySupportedByCarrier(capability)) { |
| logd("Premium capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability) |
| + " unsupported by the carrier."); |
| return false; |
| } |
| if (!isDefaultDataSub()) { |
| logd("Premium capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability) |
| + " unavailable on the non-default data subscription."); |
| return false; |
| } |
| logd("Premium capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability) |
| + " is available for purchase."); |
| return true; |
| } |
| |
| /** |
| * Purchase the given premium capability from the carrier. |
| * |
| * @param capability The premium capability to purchase. |
| * @param onComplete The callback message to send when the purchase request is complete. |
| */ |
| public synchronized void purchasePremiumCapability( |
| @TelephonyManager.PremiumCapability int capability, @NonNull Message onComplete) { |
| logd("purchasePremiumCapability: " |
| + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| // Check whether the premium capability can be purchased. |
| if (!arePremiumCapabilitiesSupportedByDevice()) { |
| sendPurchaseResult(capability, |
| TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_FEATURE_NOT_SUPPORTED, |
| onComplete); |
| return; |
| } |
| if (!isPremiumCapabilitySupportedByCarrier(capability)) { |
| sendPurchaseResult(capability, |
| PURCHASE_PREMIUM_CAPABILITY_RESULT_CARRIER_DISABLED, |
| onComplete); |
| return; |
| } |
| if (!isDefaultDataSub()) { |
| sendPurchaseResult(capability, |
| PURCHASE_PREMIUM_CAPABILITY_RESULT_NOT_DEFAULT_DATA_SUBSCRIPTION, |
| onComplete); |
| return; |
| } |
| if (isSlicingConfigActive(capability)) { |
| sendPurchaseResult(capability, |
| PURCHASE_PREMIUM_CAPABILITY_RESULT_ALREADY_PURCHASED, |
| onComplete); |
| return; |
| } |
| if (mPendingSetupCapabilities.contains(capability)) { |
| sendPurchaseResult(capability, |
| TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_PENDING_NETWORK_SETUP, |
| onComplete); |
| return; |
| } |
| if (mThrottledCapabilities.contains(capability)) { |
| sendPurchaseResult(capability, |
| TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_THROTTLED, |
| onComplete); |
| return; |
| } |
| if (!isNetworkAvailable()) { |
| sendPurchaseResult(capability, |
| TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_NETWORK_NOT_AVAILABLE, |
| onComplete); |
| return; |
| } |
| |
| if (mPendingPurchaseCapabilities.containsKey(capability)) { |
| sendPurchaseResult(capability, |
| PURCHASE_PREMIUM_CAPABILITY_RESULT_ALREADY_IN_PROGRESS, |
| onComplete); |
| return; |
| } |
| |
| // All state checks passed. Mark purchase pending and start the slice purchase application. |
| // Process through the handler since this method is synchronized. |
| mPendingPurchaseCapabilities.put(capability, onComplete); |
| sendMessage(obtainMessage(EVENT_START_SLICE_PURCHASE_APP, capability)); |
| } |
| |
| private void sendPurchaseResult(@TelephonyManager.PremiumCapability int capability, |
| @TelephonyManager.PurchasePremiumCapabilityResult int result, |
| @NonNull Message onComplete) { |
| // Send the onComplete message with the purchase result. |
| logd("Purchase result for capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability) |
| + ": " + TelephonyManager.convertPurchaseResultToString(result)); |
| AsyncResult.forMessage(onComplete, result, null); |
| onComplete.sendToTarget(); |
| } |
| |
| private void handlePurchaseResult( |
| @TelephonyManager.PremiumCapability int capability, |
| @TelephonyManager.PurchasePremiumCapabilityResult int result, boolean throttle) { |
| SlicePurchaseControllerBroadcastReceiver receiver = |
| mSlicePurchaseControllerBroadcastReceivers.remove(capability); |
| if (receiver != null) { |
| mPhone.getContext().unregisterReceiver(receiver); |
| } |
| removeMessages(EVENT_PURCHASE_TIMEOUT, capability); |
| if (throttle) { |
| throttleCapability(capability, getThrottleDuration(result)); |
| } |
| sendPurchaseResult(capability, result, mPendingPurchaseCapabilities.remove(capability)); |
| } |
| |
| private void throttleCapability(@TelephonyManager.PremiumCapability int capability, |
| long throttleDuration) { |
| // Throttle subsequent requests if necessary. |
| if (!mThrottledCapabilities.contains(capability)) { |
| if (throttleDuration > 0) { |
| logd("Throttle purchase requests for capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability) + " for " |
| + TimeUnit.MILLISECONDS.toMinutes(throttleDuration) + " minutes."); |
| mThrottledCapabilities.add(capability); |
| sendMessageDelayed(obtainMessage(EVENT_PURCHASE_UNTHROTTLED, capability), |
| throttleDuration); |
| } |
| } else { |
| reportAnomaly(UUID_CAPABILITY_THROTTLED_TWICE, |
| TelephonyManager.convertPremiumCapabilityToString(capability) |
| + " is already throttled."); |
| } |
| } |
| |
| private void onSlicingConfigChanged() { |
| for (int capability : new int[] {TelephonyManager.PREMIUM_CAPABILITY_PRIORITIZE_LATENCY}) { |
| if (isSlicingConfigActive(capability) && hasMessages(EVENT_SETUP_TIMEOUT, capability)) { |
| logd("Successfully set up slicing configuration for " |
| + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| mPendingSetupCapabilities.remove(capability); |
| removeMessages(EVENT_SETUP_TIMEOUT, capability); |
| } |
| } |
| } |
| |
| /** |
| * @return A new PremiumNetworkEntitlementApi object. |
| */ |
| @VisibleForTesting |
| public PremiumNetworkEntitlementApi getPremiumNetworkEntitlementApi() { |
| return new PremiumNetworkEntitlementApi(mPhone, getCarrierConfigs()); |
| } |
| |
| private void onStartSlicePurchaseApplication( |
| @TelephonyManager.PremiumCapability int capability) { |
| final PremiumNetworkEntitlementApi premiumNetworkEntitlementApi = |
| getPremiumNetworkEntitlementApi(); |
| PremiumNetworkEntitlementResponse premiumNetworkEntitlementResponse = |
| premiumNetworkEntitlementApi.checkEntitlementStatus(capability); |
| |
| // invalid response for entitlement check |
| if (premiumNetworkEntitlementResponse == null) { |
| logd("Invalid response for entitlement check."); |
| handlePurchaseResult(capability, |
| PURCHASE_PREMIUM_CAPABILITY_RESULT_ENTITLEMENT_CHECK_FAILED, true); |
| return; |
| } |
| |
| if (premiumNetworkEntitlementResponse.isProvisioned()) { |
| logd("Entitlement Check: Already provisioned."); |
| handlePurchaseResult(capability, |
| PURCHASE_PREMIUM_CAPABILITY_RESULT_ALREADY_PURCHASED, true); |
| return; |
| } |
| |
| if (premiumNetworkEntitlementResponse.isProvisioningInProgress()) { |
| logd("Entitlement Check: In Progress"); |
| handlePurchaseResult(capability, |
| PURCHASE_PREMIUM_CAPABILITY_RESULT_ALREADY_IN_PROGRESS, true); |
| return; |
| } |
| |
| if (!premiumNetworkEntitlementResponse.isPremiumNetworkCapabilityAllowed()) { |
| handlePurchaseResult(capability, |
| PURCHASE_PREMIUM_CAPABILITY_RESULT_ENTITLEMENT_CHECK_FAILED, true); |
| return; |
| } |
| |
| String purchaseUrl = getPurchaseUrl(premiumNetworkEntitlementResponse); |
| String carrier = getSimOperator(); |
| if (TextUtils.isEmpty(purchaseUrl) || TextUtils.isEmpty(carrier)) { |
| handlePurchaseResult(capability, |
| PURCHASE_PREMIUM_CAPABILITY_RESULT_CARRIER_DISABLED, false); |
| return; |
| } |
| |
| updateNotificationCounts(); |
| if (mMonthlyCount >= getCarrierConfigs().getInt( |
| CarrierConfigManager.KEY_PREMIUM_CAPABILITY_MAXIMUM_MONTHLY_NOTIFICATION_COUNT_INT) |
| || mDailyCount >= getCarrierConfigs().getInt( |
| CarrierConfigManager.KEY_PREMIUM_CAPABILITY_MAXIMUM_DAILY_NOTIFICATION_COUNT_INT)) { |
| logd("Reached maximum number of performance boost notifications."); |
| handlePurchaseResult(capability, |
| TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_THROTTLED, false); |
| return; |
| } |
| |
| // Start timeout for purchase completion. |
| long timeout = getCarrierConfigs().getLong(CarrierConfigManager |
| .KEY_PREMIUM_CAPABILITY_NOTIFICATION_DISPLAY_TIMEOUT_MILLIS_LONG); |
| logd("Start purchase timeout for " |
| + TelephonyManager.convertPremiumCapabilityToString(capability) + " for " |
| + TimeUnit.MILLISECONDS.toMinutes(timeout) + " minutes."); |
| sendMessageDelayed(obtainMessage(EVENT_PURCHASE_TIMEOUT, capability), timeout); |
| |
| // Broadcast start intent to start the slice purchase application |
| Intent intent = new Intent(ACTION_START_SLICE_PURCHASE_APP); |
| intent.setComponent(SLICE_PURCHASE_APP_COMPONENT_NAME); |
| intent.putExtra(EXTRA_PHONE_ID, mPhone.getPhoneId()); |
| intent.putExtra(EXTRA_SUB_ID, mPhone.getSubId()); |
| intent.putExtra(EXTRA_PREMIUM_CAPABILITY, capability); |
| intent.putExtra(EXTRA_PURCHASE_URL, purchaseUrl); |
| intent.putExtra(EXTRA_CARRIER, carrier); |
| intent.putExtra(EXTRA_INTENT_CANCELED, createPendingIntent( |
| ACTION_SLICE_PURCHASE_APP_RESPONSE_CANCELED, capability, false)); |
| intent.putExtra(EXTRA_INTENT_CARRIER_ERROR, createPendingIntent( |
| ACTION_SLICE_PURCHASE_APP_RESPONSE_CARRIER_ERROR, capability, true)); |
| intent.putExtra(EXTRA_INTENT_REQUEST_FAILED, createPendingIntent( |
| ACTION_SLICE_PURCHASE_APP_RESPONSE_REQUEST_FAILED, capability, false)); |
| intent.putExtra(EXTRA_INTENT_NOT_DEFAULT_DATA_SUBSCRIPTION, createPendingIntent( |
| ACTION_SLICE_PURCHASE_APP_RESPONSE_NOT_DEFAULT_DATA_SUBSCRIPTION, capability, |
| false)); |
| intent.putExtra(EXTRA_INTENT_SUCCESS, createPendingIntent( |
| ACTION_SLICE_PURCHASE_APP_RESPONSE_SUCCESS, capability, true)); |
| intent.putExtra(EXTRA_INTENT_NOTIFICATION_SHOWN, createPendingIntent( |
| ACTION_SLICE_PURCHASE_APP_RESPONSE_NOTIFICATION_SHOWN, capability, false)); |
| logd("Broadcasting start intent to SlicePurchaseBroadcastReceiver."); |
| mPhone.getContext().sendBroadcast(intent); |
| |
| // Listen for responses from the slice purchase application |
| mSlicePurchaseControllerBroadcastReceivers.put(capability, |
| new SlicePurchaseControllerBroadcastReceiver(capability)); |
| IntentFilter filter = new IntentFilter(); |
| filter.addAction(ACTION_SLICE_PURCHASE_APP_RESPONSE_CANCELED); |
| filter.addAction(ACTION_SLICE_PURCHASE_APP_RESPONSE_CARRIER_ERROR); |
| filter.addAction(ACTION_SLICE_PURCHASE_APP_RESPONSE_REQUEST_FAILED); |
| filter.addAction(ACTION_SLICE_PURCHASE_APP_RESPONSE_NOT_DEFAULT_DATA_SUBSCRIPTION); |
| filter.addAction(ACTION_SLICE_PURCHASE_APP_RESPONSE_SUCCESS); |
| filter.addAction(ACTION_SLICE_PURCHASE_APP_RESPONSE_NOTIFICATION_SHOWN); |
| mPhone.getContext().registerReceiver( |
| mSlicePurchaseControllerBroadcastReceivers.get(capability), filter, |
| Context.RECEIVER_NOT_EXPORTED); |
| } |
| |
| /** |
| * Get a valid purchase URL from either entitlement response or carrier configs, if one exists. |
| * |
| * @param entitlementResponse The entitlement response to get the purchase URL from. |
| * @return A valid purchase URL or an empty string if one doesn't exist. |
| */ |
| @VisibleForTesting |
| @NonNull public String getPurchaseUrl( |
| @NonNull PremiumNetworkEntitlementResponse entitlementResponse) { |
| String purchaseUrl = entitlementResponse.mServiceFlowURL; |
| if (!isUrlValid(purchaseUrl)) { |
| purchaseUrl = getCarrierConfigs().getString( |
| CarrierConfigManager.KEY_PREMIUM_CAPABILITY_PURCHASE_URL_STRING); |
| if (!isUrlValid(purchaseUrl)) { |
| purchaseUrl = ""; |
| } |
| } |
| return purchaseUrl; |
| } |
| |
| /** |
| * Get the SIM operator. This is the carrier name from the SIM rather than from the network, |
| * which will be the same regardless of whether the user is roaming or not. |
| * |
| * @return The operator name from the SIM. |
| */ |
| @VisibleForTesting |
| @Nullable public String getSimOperator() { |
| if (mPhone.getPhoneId() < TelephonyProperties.icc_operator_alpha().size()) { |
| return TelephonyProperties.icc_operator_alpha().get(mPhone.getPhoneId()); |
| } |
| return null; |
| } |
| |
| /** |
| * Create the PendingIntent to allow the slice purchase application to send back responses. |
| * |
| * @param action The action that will be sent for this PendingIntent |
| * @param capability The premium capability that was requested. |
| * @param mutable {@code true} if the PendingIntent should be mutable and |
| * {@code false} if it should be immutable. |
| * @return The PendingIntent for the given action and capability. |
| */ |
| @VisibleForTesting |
| @NonNull public PendingIntent createPendingIntent(@NonNull String action, |
| @TelephonyManager.PremiumCapability int capability, boolean mutable) { |
| Intent intent = new Intent(action); |
| intent.putExtra(EXTRA_PHONE_ID, mPhone.getPhoneId()); |
| intent.putExtra(EXTRA_PREMIUM_CAPABILITY, capability); |
| intent.setPackage(mPhone.getContext().getPackageName()); |
| return PendingIntent.getBroadcast(mPhone.getContext(), capability, intent, |
| PendingIntent.FLAG_CANCEL_CURRENT |
| | (mutable ? PendingIntent.FLAG_MUTABLE : PendingIntent.FLAG_IMMUTABLE)); |
| } |
| |
| private void onTimeout(@TelephonyManager.PremiumCapability int capability) { |
| logd("onTimeout: " + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| // Broadcast timeout intent to clean up the slice purchase notification and activity |
| Intent intent = new Intent(ACTION_SLICE_PURCHASE_APP_RESPONSE_TIMEOUT); |
| intent.setComponent(SLICE_PURCHASE_APP_COMPONENT_NAME); |
| intent.putExtra(EXTRA_PHONE_ID, mPhone.getPhoneId()); |
| intent.putExtra(EXTRA_PREMIUM_CAPABILITY, capability); |
| logd("Broadcasting timeout intent to SlicePurchaseBroadcastReceiver."); |
| mPhone.getContext().sendBroadcast(intent); |
| |
| handlePurchaseResult( |
| capability, TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_TIMEOUT, true); |
| } |
| |
| private void onCarrierError(@TelephonyManager.PremiumCapability int capability, |
| @FailureCode int failureCode, @Nullable String failureReason) { |
| logd("Carrier error for capability: " |
| + TelephonyManager.convertPremiumCapabilityToString(capability) + " with code: " |
| + convertFailureCodeToString(failureCode) + " and reason: " + failureReason); |
| if (failureCode == FAILURE_CODE_UNKNOWN && !TextUtils.isEmpty(failureReason)) { |
| reportAnomaly(UUID_UNKNOWN_FAILURE_CODE, |
| "Failure code needs to be added for: " + failureReason); |
| } |
| handlePurchaseResult(capability, |
| TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_CARRIER_ERROR, true); |
| } |
| |
| private void onCarrierSuccess(@TelephonyManager.PremiumCapability int capability, |
| long duration) { |
| logd("Successfully purchased premium capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability) + (duration > 0 |
| ? " for " + TimeUnit.MILLISECONDS.toMinutes(duration) + " minutes." : ".")); |
| mPendingSetupCapabilities.add(capability); |
| long setupDuration = getCarrierConfigs().getLong( |
| CarrierConfigManager.KEY_PREMIUM_CAPABILITY_NETWORK_SETUP_TIME_MILLIS_LONG); |
| logd("Waiting " + TimeUnit.MILLISECONDS.toMinutes(setupDuration) + " minutes for the " |
| + "network to set up the slicing configuration."); |
| sendMessageDelayed(obtainMessage(EVENT_SETUP_TIMEOUT, capability), setupDuration); |
| handlePurchaseResult( |
| capability, TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_SUCCESS, false); |
| } |
| |
| private void onSetupTimeout(@TelephonyManager.PremiumCapability int capability) { |
| logd("onSetupTimeout: " + TelephonyManager.convertPremiumCapabilityToString(capability)); |
| mPendingSetupCapabilities.remove(capability); |
| if (!isSlicingConfigActive(capability)) { |
| reportAnomaly(UUID_NETWORK_SETUP_FAILED, |
| "Failed to set up slicing configuration for capability " |
| + TelephonyManager.convertPremiumCapabilityToString(capability) |
| + " within the time specified."); |
| } |
| } |
| |
| private void onNotificationShown() { |
| SharedPreferences sp = mPhone.getContext().getSharedPreferences( |
| PERFORMANCE_BOOST_NOTIFICATION_PREFERENCES, 0); |
| mDailyCount = sp.getInt((KEY_DAILY_NOTIFICATION_COUNT + mPhone.getPhoneId()), 0) + 1; |
| mMonthlyCount = sp.getInt((KEY_MONTHLY_NOTIFICATION_COUNT + mPhone.getPhoneId()), 0) + 1; |
| logd("Performance boost notification was shown " + mDailyCount + " times today and " |
| + mMonthlyCount + " times this month."); |
| |
| SharedPreferences.Editor editor = sp.edit(); |
| editor.putInt((KEY_DAILY_NOTIFICATION_COUNT + mPhone.getPhoneId()), mDailyCount); |
| editor.putInt((KEY_MONTHLY_NOTIFICATION_COUNT + mPhone.getPhoneId()), mMonthlyCount); |
| editor.apply(); |
| |
| // Don't call updateNotificationCounts here because it will be called whenever a new |
| // purchase request comes in or when SlicePurchaseController is initialized. |
| } |
| |
| /** |
| * Update the current daily and monthly performance boost notification counts. |
| * If it has been at least a day since the last daily reset or at least a month since the last |
| * monthly reset, reset the current daily or monthly notification counts. |
| */ |
| @VisibleForTesting |
| public void updateNotificationCounts() { |
| SharedPreferences sp = mPhone.getContext().getSharedPreferences( |
| PERFORMANCE_BOOST_NOTIFICATION_PREFERENCES, 0); |
| mDailyCount = sp.getInt((KEY_DAILY_NOTIFICATION_COUNT + mPhone.getPhoneId()), 0); |
| mMonthlyCount = sp.getInt((KEY_MONTHLY_NOTIFICATION_COUNT + mPhone.getPhoneId()), 0); |
| |
| if (mLocalDate == null) { |
| // Standardize to UTC to prevent default time zone dependency |
| mLocalDate = LocalDate.now(ZoneId.of("UTC")); |
| } |
| LocalDate lastLocalDate = LocalDate.of(1, 1, 1); |
| String lastLocalDateString = sp.getString( |
| (KEY_NOTIFICATION_COUNT_LAST_RESET_DATE + mPhone.getPhoneId()), ""); |
| if (!TextUtils.isEmpty(lastLocalDateString)) { |
| try { |
| lastLocalDate = LocalDate.parse(lastLocalDateString); |
| } catch (DateTimeParseException e) { |
| loge("Error parsing LocalDate from SharedPreferences: " + e); |
| } |
| } |
| logd("updateNotificationCounts: mDailyCount=" + mDailyCount + ", mMonthlyCount=" |
| + mMonthlyCount + ", mLocalDate=" + mLocalDate + ", lastLocalDate=" |
| + lastLocalDate); |
| |
| boolean resetMonthly = lastLocalDate.getYear() != mLocalDate.getYear() |
| || lastLocalDate.getMonthValue() != mLocalDate.getMonthValue(); |
| boolean resetDaily = resetMonthly |
| || lastLocalDate.getDayOfMonth() != mLocalDate.getDayOfMonth(); |
| if (resetDaily) { |
| logd("Resetting daily" + (resetMonthly ? " and monthly" : "") + " notification count."); |
| SharedPreferences.Editor editor = sp.edit(); |
| if (resetMonthly) { |
| mMonthlyCount = 0; |
| editor.putInt((KEY_MONTHLY_NOTIFICATION_COUNT + mPhone.getPhoneId()), |
| mMonthlyCount); |
| } |
| mDailyCount = 0; |
| editor.putInt((KEY_DAILY_NOTIFICATION_COUNT + mPhone.getPhoneId()), mDailyCount); |
| editor.putString((KEY_NOTIFICATION_COUNT_LAST_RESET_DATE + mPhone.getPhoneId()), |
| mLocalDate.toString()); |
| editor.apply(); |
| } |
| } |
| |
| @Nullable private PersistableBundle getCarrierConfigs() { |
| return mPhone.getContext().getSystemService(CarrierConfigManager.class) |
| .getConfigForSubId(mPhone.getSubId()); |
| } |
| |
| private long getThrottleDuration(@TelephonyManager.PurchasePremiumCapabilityResult int result) { |
| if (result == TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_USER_CANCELED |
| || result == TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_TIMEOUT) { |
| return getCarrierConfigs().getLong(CarrierConfigManager |
| .KEY_PREMIUM_CAPABILITY_NOTIFICATION_BACKOFF_HYSTERESIS_TIME_MILLIS_LONG); |
| } |
| if (result == TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_ENTITLEMENT_CHECK_FAILED |
| || result == TelephonyManager.PURCHASE_PREMIUM_CAPABILITY_RESULT_CARRIER_ERROR) { |
| return getCarrierConfigs().getLong(CarrierConfigManager |
| .KEY_PREMIUM_CAPABILITY_PURCHASE_CONDITION_BACKOFF_HYSTERESIS_TIME_MILLIS_LONG); |
| } |
| return 0; |
| } |
| |
| private boolean isPremiumCapabilitySupportedByCarrier( |
| @TelephonyManager.PremiumCapability int capability) { |
| int[] supportedCapabilities = getCarrierConfigs().getIntArray( |
| CarrierConfigManager.KEY_SUPPORTED_PREMIUM_CAPABILITIES_INT_ARRAY); |
| if (supportedCapabilities == null) { |
| logd("No premium capabilities are supported by the carrier."); |
| return false; |
| } |
| return Arrays.stream(supportedCapabilities) |
| .anyMatch(supportedCapability -> supportedCapability == capability); |
| } |
| |
| private boolean isUrlValid(@Nullable String url) { |
| if (!URLUtil.isValidUrl(url)) { |
| loge("Invalid URL: " + url); |
| return false; |
| } |
| if (URLUtil.isAssetUrl(url) && !url.equals(SLICE_PURCHASE_TEST_FILE)) { |
| loge("Invalid asset: " + url); |
| return false; |
| } |
| try { |
| new URL(url).toURI(); |
| } catch (MalformedURLException | URISyntaxException e) { |
| loge("Invalid URI: " + url); |
| return false; |
| } |
| logd("Valid URL: " + url); |
| return true; |
| } |
| |
| private boolean arePremiumCapabilitiesSupportedByDevice() { |
| if ((mPhone.getCachedAllowedNetworkTypesBitmask() |
| & TelephonyManager.NETWORK_TYPE_BITMASK_NR) == 0) { |
| logd("Premium capabilities unsupported because NR is not allowed on the device."); |
| return false; |
| } |
| if (!mIsSlicingUpsellEnabled) { |
| logd("Premium capabilities unsupported because " |
| + "slicing upsell is disabled on the device."); |
| } |
| return mIsSlicingUpsellEnabled; |
| } |
| |
| private boolean isDefaultDataSub() { |
| return mPhone.getSubId() == SubscriptionManager.getDefaultDataSubscriptionId(); |
| } |
| |
| private boolean isSlicingConfigActive(@TelephonyManager.PremiumCapability int capability) { |
| if (mSlicingConfig == null) { |
| return false; |
| } |
| int capabilityServiceType = getSliceServiceType(capability); |
| for (NetworkSliceInfo sliceInfo : mSlicingConfig.getSliceInfo()) { |
| if (sliceInfo.getSliceServiceType() == capabilityServiceType |
| && sliceInfo.getStatus() == NetworkSliceInfo.SLICE_STATUS_ALLOWED) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| @NetworkSliceInfo.SliceServiceType private int getSliceServiceType( |
| @TelephonyManager.PremiumCapability int capability) { |
| if (capability == TelephonyManager.PREMIUM_CAPABILITY_PRIORITIZE_LATENCY) { |
| return NetworkSliceInfo.SLICE_SERVICE_TYPE_URLLC; |
| } |
| return NetworkSliceInfo.SLICE_SERVICE_TYPE_NONE; |
| } |
| |
| private boolean isNetworkAvailable() { |
| if (mPhone.getServiceState().getDataRoaming()) { |
| logd("Network unavailable because device is roaming."); |
| return false; |
| } |
| |
| if (!mPhone.getDataSettingsManager().isDataEnabledForReason( |
| TelephonyManager.DATA_ENABLED_REASON_USER)) { |
| logd("Network unavailable because user data is disabled."); |
| return false; |
| } |
| |
| // TODO (b/251558673): Create a listener for data network type changed to dismiss |
| // notification and activity when the network is no longer available. |
| switch (mPhone.getServiceState().getDataNetworkType()) { |
| case TelephonyManager.NETWORK_TYPE_NR: |
| return true; |
| case TelephonyManager.NETWORK_TYPE_LTE: |
| case TelephonyManager.NETWORK_TYPE_LTE_CA: |
| return getCarrierConfigs().getBoolean( |
| CarrierConfigManager.KEY_PREMIUM_CAPABILITY_SUPPORTED_ON_LTE_BOOL); |
| } |
| return false; |
| } |
| |
| /** |
| * Returns the failure code {@link FailureCode} as a String. |
| * |
| * @param failureCode The failure code. |
| * @return The failure code as a String. |
| */ |
| @NonNull private static String convertFailureCodeToString(@FailureCode int failureCode) { |
| switch (failureCode) { |
| case FAILURE_CODE_UNKNOWN: return "UNKNOWN"; |
| case FAILURE_CODE_CARRIER_URL_UNAVAILABLE: return "CARRIER_URL_UNAVAILABLE"; |
| case FAILURE_CODE_SERVER_UNREACHABLE: return "SERVER_UNREACHABLE"; |
| case FAILURE_CODE_AUTHENTICATION_FAILED: return "AUTHENTICATION_FAILED"; |
| case FAILURE_CODE_PAYMENT_FAILED: return "PAYMENT_FAILED"; |
| default: |
| return "UNKNOWN(" + failureCode + ")"; |
| } |
| } |
| |
| private void reportAnomaly(@NonNull String uuid, @NonNull String log) { |
| loge(log); |
| AnomalyReporter.reportAnomaly(UUID.fromString(uuid), log); |
| } |
| |
| private void logd(String s) { |
| Log.d(TAG + "-" + mPhone.getPhoneId(), s); |
| } |
| |
| private void loge(String s) { |
| Log.e(TAG + "-" + mPhone.getPhoneId(), s); |
| } |
| } |