Merge "Reduce flaky of App Info tests"
diff --git a/src/com/android/settings/dashboard/profileselector/ProfileSelectStorageFragment.java b/src/com/android/settings/dashboard/profileselector/ProfileSelectStorageFragment.java
index 1473dd1..c99b169 100644
--- a/src/com/android/settings/dashboard/profileselector/ProfileSelectStorageFragment.java
+++ b/src/com/android/settings/dashboard/profileselector/ProfileSelectStorageFragment.java
@@ -17,7 +17,6 @@
 package com.android.settings.dashboard.profileselector;
 
 import android.app.Activity;
-import android.app.settings.SettingsEnums;
 import android.content.Context;
 import android.os.Bundle;
 import android.os.UserHandle;
@@ -42,6 +41,7 @@
 import com.android.settings.deviceinfo.storage.StorageSelectionPreferenceController;
 import com.android.settings.deviceinfo.storage.StorageUsageProgressBarPreferenceController;
 import com.android.settings.deviceinfo.storage.StorageUtils;
+import com.android.settingslib.core.instrumentation.Instrumentable;
 
 import java.util.ArrayList;
 import java.util.List;
@@ -323,7 +323,7 @@
 
     @Override
     public int getMetricsCategory() {
-        return SettingsEnums.SETTINGS_STORAGE_CATEGORY;
+        return Instrumentable.METRICS_CATEGORY_UNKNOWN;
     }
 
     @Override
diff --git a/src/com/android/settings/deviceinfo/StorageCategoryFragment.java b/src/com/android/settings/deviceinfo/StorageCategoryFragment.java
index d1c8029..52f453d 100644
--- a/src/com/android/settings/deviceinfo/StorageCategoryFragment.java
+++ b/src/com/android/settings/deviceinfo/StorageCategoryFragment.java
@@ -229,7 +229,8 @@
 
     @Override
     public int getMetricsCategory() {
-        return SettingsEnums.SETTINGS_STORAGE_CATEGORY;
+        return mIsWorkProfile ? SettingsEnums.SETTINGS_STORAGE_CATEGORY_WORK :
+                SettingsEnums.SETTINGS_STORAGE_CATEGORY;
     }
 
     @Override
diff --git a/tests/uitests/src/com/android/settings/biometrics2/OWNERS b/tests/uitests/src/com/android/settings/biometrics2/OWNERS
new file mode 100644
index 0000000..c25e7cc
--- /dev/null
+++ b/tests/uitests/src/com/android/settings/biometrics2/OWNERS
@@ -0,0 +1 @@
+include /src/com/android/settings/biometrics/OWNERS
\ No newline at end of file
diff --git a/tests/uitests/src/com/android/settings/biometrics2/ui/view/FingerprintEnrollmentActivityTest.java b/tests/uitests/src/com/android/settings/biometrics2/ui/view/FingerprintEnrollmentActivityTest.java
new file mode 100644
index 0000000..24f4f89
--- /dev/null
+++ b/tests/uitests/src/com/android/settings/biometrics2/ui/view/FingerprintEnrollmentActivityTest.java
@@ -0,0 +1,118 @@
+/*
+ * 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.settings.biometrics2.ui.view;
+
+import android.content.Context;
+import android.content.Intent;
+import android.support.test.uiautomator.By;
+import android.support.test.uiautomator.UiDevice;
+import android.support.test.uiautomator.UiObject2;
+import android.support.test.uiautomator.Until;
+
+import androidx.test.InstrumentationRegistry;
+import androidx.test.runner.AndroidJUnit4;
+
+import com.android.settings.biometrics2.utils.LockScreenUtil;
+
+import org.junit.After;
+import org.junit.Assert;
+import org.junit.Before;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+
+@RunWith(AndroidJUnit4.class)
+public class FingerprintEnrollmentActivityTest {
+
+    private static final String SETTINGS_PACKAGE_NAME = "com.android.settings";
+    private static final String ACTIVITY_CLASS_NAME =
+            "com.android.settings.biometrics2.ui.view.FingerprintEnrollmentActivity";
+    private static final String EXTRA_FROM_SETTINGS_SUMMARY = "from_settings_summary";
+    private static final String EXTRA_PAGE_TRANSITION_TYPE = "page_transition_type";
+    private static final String EXTRA_KEY_CHALLENGE_TOKEN = "hw_auth_token";
+
+    private UiDevice mDevice;
+    private byte[] mToken = new byte[]{};
+    private Context mContext;
+
+    private static final int IDLE_TIMEOUT = 10000;
+
+    @Before
+    public void setUp() {
+        mDevice = UiDevice.getInstance(InstrumentationRegistry.getInstrumentation());
+        mContext = InstrumentationRegistry.getContext();
+        mDevice.pressHome();
+    }
+
+    @Test
+    public void lunchWithoutCredential() {
+        launchFingerprintEnrollActivity(true);
+        Assert.assertNotNull(mDevice.wait(Until.hasObject(
+                By.text("Choose your backup screen lock method")), IDLE_TIMEOUT));
+    }
+
+    @Test
+    public void lunchWithCredential() {
+        LockScreenUtil.setLockscreen(LockScreenUtil.LockscreenType.PIN, "1234", true);
+        launchFingerprintEnrollActivity(true);
+        Assert.assertNotNull(mDevice.wait(Until.hasObject(
+                By.text("More")), IDLE_TIMEOUT));
+
+        //click more btn twice and the introduction should stay in the last page
+        UiObject2 moreBtn = mDevice.findObject(By.text("More"));
+        moreBtn.click();
+        Assert.assertNotNull(mDevice.wait(Until.hasObject(
+                By.text("More")), IDLE_TIMEOUT));
+        moreBtn = mDevice.findObject(By.text("More"));
+        moreBtn.click();
+
+        Assert.assertNotNull(mDevice.wait(Until.hasObject(
+                By.text("I agree")), IDLE_TIMEOUT));
+        Assert.assertNotNull(mDevice.wait(Until.hasObject(
+                By.text("No thanks")), IDLE_TIMEOUT));
+
+        LockScreenUtil.resetLockscreen("1234");
+    }
+
+    @Test
+    public void launchCheckPin() {
+        LockScreenUtil.setLockscreen(LockScreenUtil.LockscreenType.PIN, "1234", true);
+        launchFingerprintEnrollActivity(false);
+        Assert.assertNotNull(mDevice.wait(Until.hasObject(
+                By.text("Enter your device PIN to continue")), IDLE_TIMEOUT));
+        LockScreenUtil.resetLockscreen("1234");
+    }
+
+    @After
+    public void tearDown() throws Exception {
+        LockScreenUtil.resetLockscreen("1234");
+        mDevice.pressHome();
+    }
+
+    private void launchFingerprintEnrollActivity(boolean hasToken) {
+        Intent intent = new Intent();
+        intent.setClassName(SETTINGS_PACKAGE_NAME, ACTIVITY_CLASS_NAME);
+        intent.putExtra(EXTRA_FROM_SETTINGS_SUMMARY, true);
+        intent.putExtra(EXTRA_PAGE_TRANSITION_TYPE, 1);
+        intent.putExtra(Intent.EXTRA_USER_ID, mContext.getUserId());
+        if (hasToken) {
+            intent.putExtra(EXTRA_KEY_CHALLENGE_TOKEN, mToken);
+        }
+        intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_NEW_TASK);
+        mContext.startActivity(intent);
+    }
+
+}
diff --git a/tests/uitests/src/com/android/settings/biometrics2/utils/LockScreenUtil.java b/tests/uitests/src/com/android/settings/biometrics2/utils/LockScreenUtil.java
new file mode 100644
index 0000000..23606e5
--- /dev/null
+++ b/tests/uitests/src/com/android/settings/biometrics2/utils/LockScreenUtil.java
@@ -0,0 +1,183 @@
+/*
+ * 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.settings.biometrics2.utils;
+
+
+import static java.lang.String.format;
+
+import android.app.KeyguardManager;
+import android.content.Context;
+import android.os.SystemClock;
+import android.support.test.uiautomator.UiDevice;
+import android.util.Log;
+
+import androidx.test.InstrumentationRegistry;
+
+import org.junit.Assert;
+
+import java.io.IOException;
+import java.util.function.Supplier;
+
+public class LockScreenUtil {
+
+    private static final String TAG = LockScreenUtil.class.getSimpleName();
+
+    private static final int SLEEP_MS = 100;
+    private static final int WAIT_TIME_MS = 10000;
+
+    private static final String SET_PATTERN_COMMAND = "locksettings set-pattern";
+    private static final String SET_PASSWORD_COMMAND = "locksettings set-password";
+    private static final String SET_PIN_COMMAND = "locksettings set-pin";
+
+    private static final String RESET_LOCKSCREEN_SHELL_COMMAND = "locksettings clear --old";
+
+    /**
+     * Different way to set the Lockscreen for Android device. Currently we only support PIN,
+     * PATTERN and PASSWORD
+     *
+     * @param lockscreenType it enum with list of supported lockscreen type
+     * @param lockscreenCode code[PIN or PATTERN or PASSWORD] which needs to be set.
+     * @param expectedResult expected result after setting the lockscreen because for lock type
+     *                       Swipe and None Keygaurd#isKeyguardSecure remain unlocked i.e. false
+     */
+    public static void setLockscreen(LockscreenType lockscreenType, String lockscreenCode,
+            boolean expectedResult) {
+        Log.d(TAG, format("Setting Lockscreen [%s(%s)]", lockscreenType, lockscreenCode));
+        switch (lockscreenType) {
+            case PIN:
+                executeShellCommand(format("%s %s", SET_PIN_COMMAND, lockscreenCode));
+                break;
+            case PASSWORD:
+                executeShellCommand(format("%s %s", SET_PASSWORD_COMMAND, lockscreenCode));
+                break;
+            case PATTERN:
+                executeShellCommand(format("%s %s", SET_PATTERN_COMMAND, lockscreenCode));
+                break;
+            default:
+                throw new AssertionError("Non-supported Lockscreen Type: " + lockscreenType);
+        }
+        assertKeyguardSecure(expectedResult);
+    }
+
+    /**
+     * Resets the give lockscreen.
+     *
+     * @param lockscreenCode old code which is currently set.
+     */
+    public static void resetLockscreen(String lockscreenCode) {
+        Log.d(TAG, String.format("Re-Setting Lockscreen %s", lockscreenCode));
+        executeShellCommand(
+                format("%s %s", RESET_LOCKSCREEN_SHELL_COMMAND, lockscreenCode));
+        assertKeyguardSecure(/* expectedSecure= */ false);
+    }
+
+
+    /**
+     * This method help you execute you shell command.
+     * Example: adb shell pm list packages -f
+     * Here you just need to provide executeShellCommand("pm list packages -f")
+     *
+     * @param command command need to executed.
+     */
+    private static void executeShellCommand(String command) {
+        Log.d(TAG, format("Executing Shell Command: %s", command));
+        try {
+            getUiDevice().executeShellCommand(command);
+        } catch (IOException e) {
+            Log.d(TAG, format("IOException Occurred: %s", e));
+        }
+    }
+
+    /**
+     * Enum for different types of Lockscreen, PIN, PATTERN and PASSWORD.
+     */
+    public enum LockscreenType {
+        PIN,
+        PASSWORD,
+        PATTERN
+    }
+
+    private static UiDevice getUiDevice() {
+        return UiDevice.getInstance(InstrumentationRegistry.getInstrumentation());
+    }
+
+    private static void assertKeyguardSecure(boolean expectedSecure) {
+        waitForCondition(
+                () -> String.format("Assert that keyguard %s secure, but failed.",
+                        expectedSecure ? "is" : "isn't"),
+                () -> getKeyguardManager().isKeyguardSecure() == expectedSecure);
+    }
+
+    /**
+     * Waits for a condition and fails if it doesn't become true within 10 sec.
+     *
+     * @param message   Supplier of the error message.
+     * @param condition Condition.
+     */
+    private static void waitForCondition(
+            Supplier<String> message, Condition condition) {
+        waitForCondition(message, condition, WAIT_TIME_MS);
+    }
+
+    /**
+     * Waits for a condition and fails if it doesn't become true within specified time period.
+     *
+     * @param message   Supplier of the error message.
+     * @param condition Condition.
+     * @param timeoutMs Timeout.
+     */
+    private static void waitForCondition(
+            Supplier<String> message, Condition condition, long timeoutMs) {
+        final long startTime = SystemClock.uptimeMillis();
+        while (SystemClock.uptimeMillis() < startTime + timeoutMs) {
+            try {
+                if (condition.isTrue()) {
+                    return;
+                }
+            } catch (Throwable t) {
+                throw new RuntimeException(t);
+            }
+            SystemClock.sleep(SLEEP_MS);
+        }
+
+        // Check once more before failing.
+        try {
+            if (condition.isTrue()) {
+                return;
+            }
+        } catch (Throwable t) {
+            throw new RuntimeException(t);
+        }
+
+        Assert.fail(message.get());
+    }
+
+    /**
+     * To get an instance of class that can be used to lock and unlock the keygaurd.
+     *
+     * @return an instance of class that can be used to lock and unlock the screen.
+     */
+    private static KeyguardManager getKeyguardManager() {
+        return (KeyguardManager) InstrumentationRegistry.getContext().getSystemService(
+                Context.KEYGUARD_SERVICE);
+    }
+
+    /** Supplier of a boolean that can throw an exception. */
+    private interface Condition {
+        boolean isTrue() throws Throwable;
+    }
+}