Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit fbac01e3 authored by android-build-team Robot's avatar android-build-team Robot Committed by Android (Google) Code Review
Browse files

Merge "New Autofill PERF test that waits for AutofillCallbacks."

parents ba720d09 984cfdf9
Loading
Loading
Loading
Loading
+47 −2
Original line number Diff line number Diff line
@@ -28,7 +28,6 @@ import android.perftests.utils.StubActivity;
import android.provider.Settings;
import android.support.test.rule.ActivityTestRule;
import android.support.test.InstrumentationRegistry;

import com.android.perftests.core.R;

import java.util.Locale;
@@ -42,12 +41,14 @@ import org.junit.ClassRule;
import org.junit.Rule;
import org.junit.runner.RunWith;

import static org.junit.Assert.assertTrue;
import static android.view.autofill.AutofillManager.AutofillCallback.EVENT_INPUT_HIDDEN;
import static android.view.autofill.AutofillManager.AutofillCallback.EVENT_INPUT_SHOWN;

public class LoginTest extends AbstractAutofillPerfTestCase {

    private EditText mUsername;
    private EditText mPassword;
    private AutofillManager mAfm;

    public LoginTest() {
        super(R.layout.test_autofill_login);
@@ -58,6 +59,7 @@ public class LoginTest extends AbstractAutofillPerfTestCase {
        View root = activity.getWindow().getDecorView();
        mUsername = root.findViewById(R.id.username);
        mPassword = root.findViewById(R.id.password);
        mAfm = activity.getSystemService(AutofillManager.class);
    }

    /**
@@ -214,4 +216,47 @@ public class LoginTest extends AbstractAutofillPerfTestCase {
            }
        });
    }

    @Test
    public void testCallbacks() throws Throwable {
        MyAutofillService.newCannedResponse()
                .setUsername(mUsername.getAutofillId(), "user")
                .setPassword(mPassword.getAutofillId(), "pass")
                .reply();
        setService();

        MyAutofillCallback callback = new MyAutofillCallback();
        mAfm.registerCallback(callback);

        // Must first focus in a field to trigger autofill and wait for service response
        // outside the loop
        mActivityRule.runOnUiThread(() -> mUsername.requestFocus());
        MyAutofillService.getLastFillRequest();
        callback.expectEvent(mUsername, EVENT_INPUT_SHOWN);

        // Now focus on password to prepare loop state
        mActivityRule.runOnUiThread(() -> mPassword.requestFocus());
        callback.expectEvent(mUsername, EVENT_INPUT_HIDDEN);
        callback.expectEvent(mPassword, EVENT_INPUT_SHOWN);

        // NOTE: we cannot run the whole loop inside the UI thread, because the autofill callback
        // is called on it, which would cause a deadlock on expectEvent().
        try {
            BenchmarkState state = mPerfStatusReporter.getBenchmarkState();
            while (state.keepRunning()) {
                mActivityRule.runOnUiThread(() -> mUsername.requestFocus());
                callback.expectEvent(mPassword, EVENT_INPUT_HIDDEN);
                callback.expectEvent(mUsername, EVENT_INPUT_SHOWN);
                mActivityRule.runOnUiThread(() -> mPassword.requestFocus());
                callback.expectEvent(mUsername, EVENT_INPUT_HIDDEN);
                callback.expectEvent(mPassword, EVENT_INPUT_SHOWN);
            }

            // Sanity check
            callback.assertNoAsyncErrors();
            MyAutofillService.assertNoAsyncErrors();
        } finally {
            mAfm.unregisterCallback(callback);
        }
    }
}
+127 −0
Original line number Diff line number Diff line
/*
 * Copyright (C) 2018 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 android.view.autofill;

import android.view.View;
import android.view.autofill.AutofillManager.AutofillCallback;

import androidx.annotation.NonNull;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.Semaphore;
import java.util.concurrent.TimeUnit;

import static android.view.autofill.AutofillManager.AutofillCallback.EVENT_INPUT_HIDDEN;
import static android.view.autofill.AutofillManager.AutofillCallback.EVENT_INPUT_SHOWN;
import static android.view.autofill.AutofillManager.AutofillCallback.EVENT_INPUT_UNAVAILABLE;

import android.os.CancellationSignal;
import android.service.autofill.FillCallback;
import android.service.autofill.FillRequest;
import android.util.Log;

/**
 * Custom {@link AutofillCallback} used to recover events during tests.
 */
public final class MyAutofillCallback extends AutofillCallback {

    private static final String TAG = "MyAutofillCallback";
    private static final int TIMEOUT_MS = 5000;

    private final BlockingQueue<MyEvent> mEvents = new LinkedBlockingQueue<>(2);
    private final List<String> mAsyncErrors = new ArrayList<>();

    @Override
    public void onAutofillEvent(View view, int event) {
        boolean offered = false;
        try {
            offered = mEvents.offer(new MyEvent(view, event), TIMEOUT_MS, TimeUnit.MILLISECONDS);
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
        if (!offered) {
            String error = "could not offer " + toString(view, event) + " in " + TIMEOUT_MS + "ms";
            Log.e(TAG, error);
            mAsyncErrors.add(error);
        }
    }

    /**
     * Asserts the callback is called for the given view and event, or fail if it times out.
     */
    public void expectEvent(@NonNull View view, int event) {
        MyEvent myEvent;
        try {
            myEvent = mEvents.poll(TIMEOUT_MS, TimeUnit.MILLISECONDS);
            if (myEvent == null) {
                throw new IllegalStateException("no event received in " + TIMEOUT_MS
                        + "ms while waiting for " + toString(view, event));
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
            throw new IllegalStateException("interrupted waiting for " + toString(view, event));
        }
        if (!myEvent.view.equals(view) || myEvent.event != event) {
            throw new AssertionError("Invalid event: expected " + myEvent + ", got "
                    + toString(view, event));
        }
    }

    /**
     * Throws an exception if an error happened asynchronously while handing
     * {@link #onAutofillEvent(View, int)}.
     */
    public void assertNoAsyncErrors() {
       if (!mAsyncErrors.isEmpty()) {
           throw new IllegalStateException(mAsyncErrors.size() + " errors: " + mAsyncErrors);
       }
    }

    private static String eventToString(int event) {
        switch (event) {
            case EVENT_INPUT_HIDDEN:
                return "HIDDEN";
            case EVENT_INPUT_SHOWN:
                return "SHOWN";
            case EVENT_INPUT_UNAVAILABLE:
                return "UNAVAILABLE";
            default:
                throw new IllegalArgumentException("invalid event: " + event);
        }
    }

    private static String toString(View view, int event) {
        return eventToString(event) + ": " + view + ")";
    }

    private static final class MyEvent {
        public final View view;
        public final int event;

        MyEvent(View view, int event) {
            this.view = view;
            this.event = event;
        }

        @Override
        public String toString() {
            return MyAutofillCallback.toString(view, event);
        }
    }
}
+1 −1
Original line number Diff line number Diff line
@@ -69,7 +69,7 @@ public class MyAutofillService extends AutofillService {
     */
    static void assertNoAsyncErrors() {
       if (!sAsyncErrors.isEmpty()) {
           throw new IllegalStateException("got errors: " + sAsyncErrors);
           throw new IllegalStateException(sAsyncErrors.size() + " errors: " + sAsyncErrors);
       }
    }