/* * Copyright (C) 2015 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.settings; import static com.android.settingslib.RestrictedLockUtils.EnforcedAdmin; import android.app.Activity; import android.app.ProgressDialog; import android.app.settings.SettingsEnums; import android.bluetooth.BluetoothAdapter; import android.bluetooth.BluetoothManager; import android.content.ContentResolver; import android.content.Context; import android.net.ConnectivityManager; import android.net.NetworkPolicyManager; import android.net.Uri; import android.net.VpnManager; import android.net.wifi.WifiManager; import android.net.wifi.p2p.WifiP2pManager; import android.os.AsyncTask; import android.os.Bundle; import android.os.Looper; import android.os.RecoverySystem; import android.os.UserHandle; import android.os.UserManager; import android.telephony.SubscriptionManager; import android.telephony.SubscriptionManager.OnSubscriptionsChangedListener; import android.telephony.TelephonyManager; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.Button; import android.widget.TextView; import android.widget.Toast; import androidx.annotation.VisibleForTesting; import androidx.appcompat.app.AlertDialog; import com.android.settings.core.InstrumentedFragment; import com.android.settings.enterprise.ActionDisabledByAdminDialogHelper; import com.android.settings.network.apn.ApnSettings; import com.android.settingslib.RestrictedLockUtilsInternal; /** * Confirm and execute a reset of the network settings to a clean "just out of the box" * state. Multiple confirmations are required: first, a general "are you sure * you want to do this?" prompt, followed by a keyguard pattern trace if the user * has defined one, followed by a final strongly-worded "THIS WILL RESET EVERYTHING" * prompt. If at any time the phone is allowed to go to sleep, is * locked, et cetera, then the confirmation sequence is abandoned. * * This is the confirmation screen. */ public class ResetNetworkConfirm extends InstrumentedFragment { private static final String TAG = "ResetNetworkConfirm"; @VisibleForTesting View mContentView; @VisibleForTesting boolean mEraseEsim; @VisibleForTesting ResetNetworkTask mResetNetworkTask; @VisibleForTesting Activity mActivity; private int mSubId = SubscriptionManager.INVALID_SUBSCRIPTION_ID; private ProgressDialog mProgressDialog; private AlertDialog mAlertDialog; private OnSubscriptionsChangedListener mSubscriptionsChangedListener; /** * Async task used to do all reset task. If error happens during * erasing eSIM profiles or timeout, an error msg is shown. */ private class ResetNetworkTask extends AsyncTask { private static final String TAG = "ResetNetworkTask"; private final Context mContext; private final String mPackageName; ResetNetworkTask(Context context) { mContext = context; mPackageName = context.getPackageName(); } @Override protected Boolean doInBackground(Void... params) { boolean isResetSucceed = true; ConnectivityManager connectivityManager = (ConnectivityManager) mContext.getSystemService(Context.CONNECTIVITY_SERVICE); if (connectivityManager != null) { connectivityManager.factoryReset(); } VpnManager vpnManager = mContext.getSystemService(VpnManager.class); if (vpnManager != null) { vpnManager.factoryReset(); } WifiManager wifiManager = (WifiManager) mContext.getSystemService(Context.WIFI_SERVICE); if (wifiManager != null) { wifiManager.factoryReset(); } p2pFactoryReset(mContext); if (mEraseEsim) { isResetSucceed = RecoverySystem.wipeEuiccData(mContext, mPackageName); } TelephonyManager telephonyManager = (TelephonyManager) mContext.getSystemService(TelephonyManager.class) .createForSubscriptionId(mSubId); if (telephonyManager != null) { telephonyManager.resetSettings(); } NetworkPolicyManager policyManager = (NetworkPolicyManager) mContext.getSystemService(Context.NETWORK_POLICY_SERVICE); if (policyManager != null) { String subscriberId = telephonyManager.getSubscriberId(); policyManager.factoryReset(subscriberId); } BluetoothManager btManager = (BluetoothManager) mContext.getSystemService(Context.BLUETOOTH_SERVICE); if (btManager != null) { BluetoothAdapter btAdapter = btManager.getAdapter(); if (btAdapter != null) { btAdapter.factoryReset(); } } restoreDefaultApn(mContext); Log.d(TAG, "network factoryReset complete. succeeded: " + String.valueOf(isResetSucceed)); return isResetSucceed; } @Override protected void onPostExecute(Boolean succeeded) { mProgressDialog.dismiss(); if (succeeded) { Toast.makeText(mContext, R.string.reset_network_complete_toast, Toast.LENGTH_SHORT) .show(); } else { mAlertDialog = new AlertDialog.Builder(mContext) .setTitle(R.string.reset_esim_error_title) .setMessage(R.string.reset_esim_error_msg) .setPositiveButton(android.R.string.ok, null /* listener */) .show(); } } } /** * The user has gone through the multiple confirmation, so now we go ahead * and reset the network settings to its factory-default state. */ @VisibleForTesting Button.OnClickListener mFinalClickListener = new Button.OnClickListener() { @Override public void onClick(View v) { if (Utils.isMonkeyRunning()) { return; } // abandon execution if subscription no longer active if (mSubId != SubscriptionManager.INVALID_SUBSCRIPTION_ID) { SubscriptionManager mgr = getSubscriptionManager(); // always remove listener stopMonitorSubscriptionChange(mgr); if (!isSubscriptionRemainActive(mgr, mSubId)) { Log.w(TAG, "subId " + mSubId + " disappear when confirm"); mActivity.finish(); return; } } mProgressDialog = getProgressDialog(mActivity); mProgressDialog.show(); mResetNetworkTask = new ResetNetworkTask(mActivity); mResetNetworkTask.execute(); } }; @VisibleForTesting void p2pFactoryReset(Context context) { WifiP2pManager wifiP2pManager = (WifiP2pManager) context.getSystemService(Context.WIFI_P2P_SERVICE); if (wifiP2pManager != null) { WifiP2pManager.Channel channel = wifiP2pManager.initialize( context.getApplicationContext(), context.getMainLooper(), null /* listener */); if (channel != null) { wifiP2pManager.factoryReset(channel, null /* listener */); } } } private ProgressDialog getProgressDialog(Context context) { final ProgressDialog progressDialog = new ProgressDialog(context); progressDialog.setIndeterminate(true); progressDialog.setCancelable(false); progressDialog.setMessage( context.getString(R.string.main_clear_progress_text)); return progressDialog; } /** * Restore APN settings to default. */ private void restoreDefaultApn(Context context) { Uri uri = Uri.parse(ApnSettings.RESTORE_CARRIERS_URI); if (SubscriptionManager.isUsableSubscriptionId(mSubId)) { uri = Uri.withAppendedPath(uri, "subId/" + String.valueOf(mSubId)); } ContentResolver resolver = context.getContentResolver(); resolver.delete(uri, null, null); } /** * Configure the UI for the final confirmation interaction */ private void establishFinalConfirmationState() { mContentView.findViewById(R.id.execute_reset_network) .setOnClickListener(mFinalClickListener); } @VisibleForTesting void setSubtitle() { if (mEraseEsim) { ((TextView) mContentView.findViewById(R.id.reset_network_confirm)) .setText(R.string.reset_network_final_desc_esim); } } @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { final EnforcedAdmin admin = RestrictedLockUtilsInternal.checkIfRestrictionEnforced( mActivity, UserManager.DISALLOW_NETWORK_RESET, UserHandle.myUserId()); if (RestrictedLockUtilsInternal.hasBaseUserRestriction(mActivity, UserManager.DISALLOW_NETWORK_RESET, UserHandle.myUserId())) { return inflater.inflate(R.layout.network_reset_disallowed_screen, null); } else if (admin != null) { new ActionDisabledByAdminDialogHelper(mActivity) .prepareDialogBuilder(UserManager.DISALLOW_NETWORK_RESET, admin) .setOnDismissListener(__ -> mActivity.finish()) .show(); return new View(mActivity); } mContentView = inflater.inflate(R.layout.reset_network_confirm, null); establishFinalConfirmationState(); setSubtitle(); return mContentView; } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Bundle args = getArguments(); if (args != null) { mSubId = args.getInt(SubscriptionManager.EXTRA_SUBSCRIPTION_INDEX, SubscriptionManager.INVALID_SUBSCRIPTION_ID); mEraseEsim = args.getBoolean(MainClear.ERASE_ESIMS_EXTRA); } mActivity = getActivity(); if (mSubId == SubscriptionManager.INVALID_SUBSCRIPTION_ID) { return; } // close confirmation dialog when reset specific subscription // but removed priori to the confirmation button been pressed startMonitorSubscriptionChange(getSubscriptionManager()); } private SubscriptionManager getSubscriptionManager() { SubscriptionManager mgr = mActivity.getSystemService(SubscriptionManager.class); if (mgr == null) { Log.w(TAG, "No SubscriptionManager"); } return mgr; } private void startMonitorSubscriptionChange(SubscriptionManager mgr) { if (mgr == null) { return; } // update monitor listener mSubscriptionsChangedListener = new OnSubscriptionsChangedListener( Looper.getMainLooper()) { @Override public void onSubscriptionsChanged() { SubscriptionManager mgr = getSubscriptionManager(); if (isSubscriptionRemainActive(mgr, mSubId)) { return; } // close UI if subscription no longer active Log.w(TAG, "subId " + mSubId + " no longer active."); stopMonitorSubscriptionChange(mgr); mActivity.finish(); } }; mgr.addOnSubscriptionsChangedListener( mActivity.getMainExecutor(), mSubscriptionsChangedListener); } private boolean isSubscriptionRemainActive(SubscriptionManager mgr, int subscriptionId) { return (mgr == null) ? false : (mgr.getActiveSubscriptionInfo(subscriptionId) != null); } private void stopMonitorSubscriptionChange(SubscriptionManager mgr) { if ((mgr == null) || (mSubscriptionsChangedListener == null)) { return; } mgr.removeOnSubscriptionsChangedListener(mSubscriptionsChangedListener); mSubscriptionsChangedListener = null; } @Override public void onDestroy() { if (mResetNetworkTask != null) { mResetNetworkTask.cancel(true /* mayInterruptIfRunning */); mResetNetworkTask = null; } if (mProgressDialog != null) { mProgressDialog.dismiss(); } if (mAlertDialog != null) { mAlertDialog.dismiss(); } stopMonitorSubscriptionChange(getSubscriptionManager()); super.onDestroy(); } @Override public int getMetricsCategory() { return SettingsEnums.RESET_NETWORK_CONFIRM; } }