diff --git a/app/build.gradle b/app/build.gradle index b6659fd2..9ccb36a6 100644 --- a/app/build.gradle +++ b/app/build.gradle @@ -7,17 +7,17 @@ ext { // exactly 1 digit versionMinor = 0 // exactly 2 digits - versionBuild = 05 + versionBuild = 06 } android { - compileSdkVersion 26 - buildToolsVersion '26.0.0' + compileSdkVersion 27 + buildToolsVersion "27.0.1" defaultConfig { applicationId "com.afwsamples.testdpc" minSdkVersion 21 - targetSdkVersion 26 + targetSdkVersion 27 versionCode versionMajor * 1000 + versionMinor * 100 + versionBuild versionName "${versionMajor}.${versionMinor}.${versionBuild}" } @@ -61,9 +61,9 @@ android { } dependencies { - api 'com.android.support:preference-v14:26.+' - api 'com.android.support:recyclerview-v7:26.+' - api 'com.android.support:support-v13:26.+' + api 'com.android.support:preference-v14:27.0.2' + api 'com.android.support:recyclerview-v7:27.0.2' + api 'com.android.support:support-v13:27.0.2' api 'com.google.android.gms:play-services-safetynet:+' api(name:'setup-wizard-lib-platform-release', ext:'aar') } diff --git a/app/src/main/java/com/afwsamples/testdpc/DeviceAdminReceiver.java b/app/src/main/java/com/afwsamples/testdpc/DeviceAdminReceiver.java index 2840dcee..d384bb8a 100644 --- a/app/src/main/java/com/afwsamples/testdpc/DeviceAdminReceiver.java +++ b/app/src/main/java/com/afwsamples/testdpc/DeviceAdminReceiver.java @@ -17,7 +17,6 @@ package com.afwsamples.testdpc; import android.annotation.TargetApi; -import android.app.Notification; import android.app.NotificationManager; import android.app.PendingIntent; import android.app.admin.DevicePolicyManager; @@ -33,13 +32,12 @@ import android.os.UserHandle; import android.os.UserManager; import android.preference.PreferenceManager; -import android.support.v7.app.NotificationCompat; +import android.support.v4.app.NotificationCompat; import android.text.TextUtils; import android.util.Log; import android.widget.Toast; import com.afwsamples.testdpc.common.NotificationUtil; -import com.afwsamples.testdpc.common.Util; import com.afwsamples.testdpc.provision.PostProvisioningTask; import java.io.BufferedReader; diff --git a/app/src/main/java/com/afwsamples/testdpc/PackageMonitorReceiver.java b/app/src/main/java/com/afwsamples/testdpc/PackageMonitorReceiver.java index 991f1eda..6d6e35b4 100644 --- a/app/src/main/java/com/afwsamples/testdpc/PackageMonitorReceiver.java +++ b/app/src/main/java/com/afwsamples/testdpc/PackageMonitorReceiver.java @@ -5,8 +5,7 @@ import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; -import android.support.v7.app.NotificationCompat; -import android.support.v7.app.NotificationCompat.Builder; +import android.support.v4.app.NotificationCompat; import android.text.TextUtils; import com.afwsamples.testdpc.common.NotificationUtil; @@ -37,6 +36,7 @@ public void onReceive(Context context, Intent intent) { .setContentText(notificationBody) .setStyle(new NotificationCompat.BigTextStyle().bigText(notificationBody)) .setDefaults(Notification.DEFAULT_LIGHTS) + .setOnlyAlertOnce(true) .build(); NotificationManager notificationManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE); diff --git a/app/src/main/java/com/afwsamples/testdpc/common/NotificationUtil.java b/app/src/main/java/com/afwsamples/testdpc/common/NotificationUtil.java index 63845852..1ce37d9c 100644 --- a/app/src/main/java/com/afwsamples/testdpc/common/NotificationUtil.java +++ b/app/src/main/java/com/afwsamples/testdpc/common/NotificationUtil.java @@ -8,7 +8,7 @@ import android.support.annotation.RequiresApi; import android.support.annotation.StringRes; import android.support.v4.os.BuildCompat; -import android.support.v7.app.NotificationCompat; +import android.support.v4.app.NotificationCompat; import com.afwsamples.testdpc.R; @@ -37,8 +37,8 @@ public static NotificationCompat.Builder getNotificationBuilder(Context context) if (BuildCompat.isAtLeastO()) { createDefaultNotificationChannel(context); } - NotificationCompat.Builder builder = new NotificationCompat.Builder(context); - builder.setChannelId(DEFAULT_CHANNEL_ID); + NotificationCompat.Builder builder + = new NotificationCompat.Builder(context, DEFAULT_CHANNEL_ID); return builder; } diff --git a/app/src/main/java/com/afwsamples/testdpc/common/Util.java b/app/src/main/java/com/afwsamples/testdpc/common/Util.java index 1a1f4da2..0ebd9e80 100644 --- a/app/src/main/java/com/afwsamples/testdpc/common/Util.java +++ b/app/src/main/java/com/afwsamples/testdpc/common/Util.java @@ -17,9 +17,6 @@ package com.afwsamples.testdpc.common; import android.annotation.TargetApi; -import android.app.Notification; -import android.app.NotificationChannel; -import android.app.NotificationManager; import android.app.Service; import android.app.admin.DevicePolicyManager; import android.content.ActivityNotFoundException; @@ -30,12 +27,11 @@ import android.net.Uri; import android.os.Build; import android.os.Build.VERSION_CODES; +import android.os.Bundle; import android.os.UserHandle; import android.os.UserManager; -import android.support.annotation.RequiresApi; import android.support.v14.preference.PreferenceFragment; import android.support.v4.os.BuildCompat; -import android.support.v7.app.NotificationCompat; import android.text.format.DateUtils; import android.util.Log; import android.widget.ImageView; @@ -56,7 +52,12 @@ */ public class Util { private static final String TAG = "Util"; - private static final int DEFAULT_BUFFER_SIZE = 4096; + private static final int DEFAULT_BUFFER_SIZE = 4096; + + private static final String BROADCAST_ACTION_FRP_CONFIG_CHANGED = + "com.google.android.gms.auth.FRP_CONFIG_CHANGED"; + private static final String GMSCORE_PACKAGE = "com.google.android.gms"; + private static final String PERSISTENT_DEVICE_OWNER_STATE = "persistentDeviceOwnerState"; /** * Format a friendly datetime for the current locale according to device policy documentation. @@ -194,6 +195,42 @@ public static boolean installCaCertificate(InputStream certificateInputStream, return false; } + /** + * Returns the persistent device owner state which has been set by the device owner as an app + * restriction on GmsCore or null if there is no such restriction set. + */ + @TargetApi(VERSION_CODES.O) + public static String getPersistentDoStateFromApplicationRestriction( + DevicePolicyManager dpm, ComponentName admin) { + Bundle restrictions = dpm.getApplicationRestrictions(admin, GMSCORE_PACKAGE); + return restrictions.getString(PERSISTENT_DEVICE_OWNER_STATE); + } + + /** + * Sets the persistent device owner state by setting a special app restriction on GmsCore and + * notifies GmsCore about the change by sending a broadcast. + * + * @param state The device owner state to be preserved across factory resets. If null, the + * persistent device owner state and the corresponding restiction are cleared. + */ + @TargetApi(VERSION_CODES.O) + public static void setPersistentDoStateWithApplicationRestriction( + Context context, DevicePolicyManager dpm, ComponentName admin, String state) { + Bundle restrictions = dpm.getApplicationRestrictions(admin, GMSCORE_PACKAGE); + if (state == null) { + // Clear the restriction + restrictions.remove(PERSISTENT_DEVICE_OWNER_STATE); + } else { + // Set the restriction + restrictions.putString(PERSISTENT_DEVICE_OWNER_STATE, state); + } + dpm.setApplicationRestrictions(admin, GMSCORE_PACKAGE, restrictions); + Intent broadcastIntent = new Intent(BROADCAST_ACTION_FRP_CONFIG_CHANGED); + broadcastIntent.setPackage(GMSCORE_PACKAGE); + broadcastIntent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND); + context.sendBroadcast(broadcastIntent); + } + private static DevicePolicyManager getDevicePolicyManager(Context context) { return (DevicePolicyManager)context.getSystemService(Service.DEVICE_POLICY_SERVICE); } diff --git a/app/src/main/java/com/afwsamples/testdpc/common/preference/DpcPreferenceHelper.java b/app/src/main/java/com/afwsamples/testdpc/common/preference/DpcPreferenceHelper.java index f41a7f81..15c38869 100644 --- a/app/src/main/java/com/afwsamples/testdpc/common/preference/DpcPreferenceHelper.java +++ b/app/src/main/java/com/afwsamples/testdpc/common/preference/DpcPreferenceHelper.java @@ -93,9 +93,6 @@ public class DpcPreferenceHelper { * not yet assigned. */ private int getDeviceSdkInt() { - if (BuildCompat.isAtLeastO()) { - return Build.VERSION_CODES.O; - } return Build.VERSION.SDK_INT; } diff --git a/app/src/main/java/com/afwsamples/testdpc/policy/PersistentDeviceOwnerFragment.java b/app/src/main/java/com/afwsamples/testdpc/policy/PersistentDeviceOwnerFragment.java new file mode 100644 index 00000000..2d256379 --- /dev/null +++ b/app/src/main/java/com/afwsamples/testdpc/policy/PersistentDeviceOwnerFragment.java @@ -0,0 +1,77 @@ +package com.afwsamples.testdpc.policy; + +import android.annotation.TargetApi; +import android.app.Fragment; +import android.app.admin.DevicePolicyManager; +import android.content.ComponentName; +import android.content.Context; +import android.content.Intent; +import android.os.Build; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.Button; +import android.widget.EditText; + +import com.afwsamples.testdpc.DeviceAdminReceiver; +import com.afwsamples.testdpc.R; +import com.afwsamples.testdpc.common.Util; + +/** + * Allows the user to set a test persistent device owner state. + * + *
For manual testing of forced re-enrollment. + * + *
If there is a non-empty peristent device owner state, it will survive the next factory reset,
+ * TestDPC will be re-installed automatically as device owner and the state will be passed to it
+ * during the initial device setup.
+ */
+public class PersistentDeviceOwnerFragment extends Fragment implements View.OnClickListener {
+
+ private DevicePolicyManager mDpm;
+ private ComponentName mAdminComponent;
+ private EditText mStateEdit;
+
+ @Override
+ public void onCreate(Bundle savedInstanceState) {
+ super.onCreate(savedInstanceState);
+ getActivity().getActionBar().setTitle(R.string.persistent_device_owner);
+ mDpm = (DevicePolicyManager) getActivity().getSystemService(
+ Context.DEVICE_POLICY_SERVICE);
+ mAdminComponent = DeviceAdminReceiver.getComponentName(getActivity());
+ }
+
+ @Override
+ public View onCreateView(
+ LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
+ View root = inflater.inflate(R.layout.persistent_device_owner_fragment, container, false);
+ root.findViewById(R.id.clear_persistent_device_owner_button).setOnClickListener(this);
+ root.findViewById(R.id.set_persistent_device_owner_button).setOnClickListener(this);
+ mStateEdit = (EditText) root.findViewById(R.id.persistent_device_owner_state_edit);
+ return root;
+ }
+
+ @Override
+ public void onClick(View view) {
+ String message = null;
+ switch (view.getId()) {
+ case R.id.clear_persistent_device_owner_button:
+ mStateEdit.getText().clear();
+ Util.setPersistentDoStateWithApplicationRestriction(
+ getActivity(), mDpm, mAdminComponent, null);
+ break;
+ case R.id.set_persistent_device_owner_button:
+ Util.setPersistentDoStateWithApplicationRestriction(
+ getActivity(), mDpm, mAdminComponent, mStateEdit.getText().toString());
+ break;
+ }
+ }
+
+ @Override
+ public void onResume() {
+ super.onResume();
+ String state = Util.getPersistentDoStateFromApplicationRestriction(mDpm, mAdminComponent);
+ mStateEdit.setText(state == null ? "" : state);
+ }
+}
diff --git a/app/src/main/java/com/afwsamples/testdpc/policy/PolicyManagementFragment.java b/app/src/main/java/com/afwsamples/testdpc/policy/PolicyManagementFragment.java
index 06f88d13..b97b0e99 100644
--- a/app/src/main/java/com/afwsamples/testdpc/policy/PolicyManagementFragment.java
+++ b/app/src/main/java/com/afwsamples/testdpc/policy/PolicyManagementFragment.java
@@ -298,6 +298,7 @@ public class PolicyManagementFragment extends BaseSearchablePolicyPreferenceFrag
private static final String UNHIDE_APPS_KEY = "unhide_apps";
private static final String UNSUSPEND_APPS_KEY = "unsuspend_apps";
private static final String WIPE_DATA_KEY = "wipe_data";
+ private static final String PERSISTENT_DEVICE_OWNER_KEY = "persistent_device_owner";
private static final String CREATE_WIFI_CONFIGURATION_KEY = "create_wifi_configuration";
private static final String CREATE_EAP_TLS_WIFI_CONFIGURATION_KEY
= "create_eap_tls_wifi_configuration";
@@ -409,6 +410,7 @@ public void onCreatePreferences(Bundle savedInstanceState, String rootKey) {
STAY_ON_WHILE_PLUGGED_IN);
mStayOnWhilePluggedInSwitchPreference.setOnPreferenceChangeListener(this);
findPreference(WIPE_DATA_KEY).setOnPreferenceClickListener(this);
+ findPreference(PERSISTENT_DEVICE_OWNER_KEY).setOnPreferenceClickListener(this);
findPreference(REMOVE_DEVICE_OWNER_KEY).setOnPreferenceClickListener(this);
mEnableBackupServicePreference = (SwitchPreference) findPreference(ENABLE_BACKUP_SERVICE);
mEnableBackupServicePreference.setOnPreferenceChangeListener(this);
@@ -482,6 +484,10 @@ public void onCreatePreferences(Bundle savedInstanceState, String rootKey) {
SET_AUTO_TIME_REQUIRED_KEY);
mSetAutoTimeRequiredPreference.setOnPreferenceChangeListener(this);
+ mSetDeviceOrganizationNamePreference =
+ (EditTextPreference) findPreference(SET_DEVICE_ORGANIZATION_NAME_KEY);
+ mSetDeviceOrganizationNamePreference.setOnPreferenceChangeListener(this);
+
constrainSpecialCasePreferences();
maybeDisableLockTaskPreferences();
@@ -585,6 +591,9 @@ public void onPositiveButtonClicked(String[] lockTaskArray) {
case WIPE_DATA_KEY:
showWipeDataPrompt();
return true;
+ case PERSISTENT_DEVICE_OWNER_KEY:
+ showFragment(new PersistentDeviceOwnerFragment());
+ return true;
case REMOVE_DEVICE_OWNER_KEY:
showRemoveDeviceOwnerPrompt();
return true;
diff --git a/app/src/main/java/com/afwsamples/testdpc/provision/PostProvisioningTask.java b/app/src/main/java/com/afwsamples/testdpc/provision/PostProvisioningTask.java
index 67cb0946..be4de37b 100644
--- a/app/src/main/java/com/afwsamples/testdpc/provision/PostProvisioningTask.java
+++ b/app/src/main/java/com/afwsamples/testdpc/provision/PostProvisioningTask.java
@@ -37,6 +37,7 @@
import android.util.Log;
import com.afwsamples.testdpc.AddAccountActivity;
+import com.afwsamples.testdpc.DeviceAdminReceiver;
import com.afwsamples.testdpc.EnableDeviceOwnerActivity;
import com.afwsamples.testdpc.EnableProfileActivity;
import com.afwsamples.testdpc.common.LaunchIntentUtil;
@@ -66,6 +67,8 @@ public class PostProvisioningTask {
"com.afwsamples.testdpc.SetupManagementLaunchActivity";
private static final String POST_PROV_PREFS = "post_prov_prefs";
private static final String KEY_POST_PROV_DONE = "key_post_prov_done";
+ private static final String KEY_DEVICE_OWNER_STATE =
+ "android.app.extra.PERSISTENT_DEVICE_OWNER_STATE";
private final Context mContext;
private final DevicePolicyManager mDevicePolicyManager;
@@ -98,6 +101,18 @@ public boolean performPostProvisioningOperations(Intent intent) {
maybeSetAffiliationIds(extras);
}
+ // If TestDPC asked GmsCore to store its state in the FRP area before factory reset, the
+ // state will be handed over to it during the next device setup.
+ if (BuildCompat.isAtLeastOMR1()
+ && extras != null
+ && extras.containsKey(KEY_DEVICE_OWNER_STATE)) {
+ Util.setPersistentDoStateWithApplicationRestriction(
+ mContext,
+ mDevicePolicyManager,
+ DeviceAdminReceiver.getComponentName(mContext),
+ extras.getString(KEY_DEVICE_OWNER_STATE));
+ }
+
// Hide the setup launcher when this app is the admin
mContext.getPackageManager().setComponentEnabledSetting(
new ComponentName(mContext, SETUP_MANAGEMENT_LAUNCH_ACTIVITY),
diff --git a/app/src/main/res/layout/persistent_device_owner_fragment.xml b/app/src/main/res/layout/persistent_device_owner_fragment.xml
new file mode 100644
index 00000000..92543834
--- /dev/null
+++ b/app/src/main/res/layout/persistent_device_owner_fragment.xml
@@ -0,0 +1,46 @@
+
+
+