Merge "Use FakeTimer in test to avoid flakiness."

This commit is contained in:
Treehugger Robot
2023-06-16 00:10:11 +00:00
committed by Android (Google) Code Review
3 changed files with 166 additions and 36 deletions

View File

@@ -119,7 +119,7 @@ public class ScreenFlashNotificationColorDialogFragment extends DialogFragment i
synchronized (this) {
if (mTimer != null) mTimer.cancel();
mTimer = new Timer();
mTimer = createTimer();
if (mIsPreview) {
mTimer.schedule(getStopTask(), 0);
startDelay = BETWEEN_STOP_AND_START_DELAY_MS;
@@ -176,4 +176,8 @@ public class ScreenFlashNotificationColorDialogFragment extends DialogFragment i
getContext().sendBroadcast(stopIntent);
mIsPreview = false;
}
Timer createTimer() {
return new Timer();
}
}

View File

@@ -42,6 +42,7 @@ import androidx.appcompat.app.AlertDialog;
import androidx.fragment.app.FragmentActivity;
import com.android.settings.R;
import com.android.settings.testutils.FakeTimer;
import org.junit.Before;
import org.junit.Test;
@@ -49,9 +50,12 @@ import org.junit.runner.RunWith;
import org.robolectric.Robolectric;
import org.robolectric.RobolectricTestRunner;
import org.robolectric.shadows.ShadowContextWrapper;
import org.robolectric.util.ReflectionHelpers;
import java.util.ArrayList;
import java.util.List;
import java.util.Timer;
import java.util.function.Consumer;
@RunWith(RobolectricTestRunner.class)
public class ScreenFlashNotificationColorDialogFragmentTest {
@@ -68,9 +72,8 @@ public class ScreenFlashNotificationColorDialogFragmentTest {
mShadowContextWrapper = shadowOf(fragmentActivity);
mCurrentColor = ROSE.mColorInt;
mDialogFragment = ScreenFlashNotificationColorDialogFragment.getInstance(
mCurrentColor, selectedColor -> mCurrentColor = selectedColor
);
mDialogFragment = createFragment();
mDialogFragment.show(fragmentActivity.getSupportFragmentManager(), "test");
mAlertDialog = (AlertDialog) mDialogFragment.getDialog();
@@ -91,16 +94,19 @@ public class ScreenFlashNotificationColorDialogFragmentTest {
}
@Test
public void clickNeutral_assertStartPreview() throws InterruptedException {
public void clickNeutral_assertStartPreview() {
performClickOnDialog(BUTTON_NEUTRAL);
Thread.sleep(100);
getTimerFromFragment().runOneTask();
Intent captured = getLastCapturedIntent();
assertThat(captured.getAction()).isEqualTo(ACTION_FLASH_NOTIFICATION_START_PREVIEW);
assertThat(captured.getIntExtra(EXTRA_FLASH_NOTIFICATION_PREVIEW_TYPE, TYPE_SHORT_PREVIEW))
.isEqualTo(TYPE_LONG_PREVIEW);
assertThat(captured.getIntExtra(EXTRA_FLASH_NOTIFICATION_PREVIEW_COLOR, Color.TRANSPARENT))
.isEqualTo(ROSE.mColorInt);
assertStartPreview(ROSE.mColorInt);
}
@Test
public void clickNeutral_flushAllScheduledTasks_assertStopPreview() {
performClickOnDialog(BUTTON_NEUTRAL);
getTimerFromFragment().runAllTasks();
assertStopPreview();
}
@Test
@@ -116,51 +122,47 @@ public class ScreenFlashNotificationColorDialogFragmentTest {
}
@Test
public void clickNeutralAndPause_assertStopPreview() throws InterruptedException {
public void clickNeutralAndPause_assertStopPreview() {
performClickOnDialog(BUTTON_NEUTRAL);
Thread.sleep(100);
getTimerFromFragment().runOneTask();
mDialogFragment.onPause();
Thread.sleep(100);
assertThat(getLastCapturedIntent().getAction())
.isEqualTo(ACTION_FLASH_NOTIFICATION_STOP_PREVIEW);
assertStopPreview();
}
@Test
public void clickNeutralAndClickNegative_assertStopPreview() throws InterruptedException {
public void clickNeutralAndClickNegative_assertStopPreview() {
performClickOnDialog(BUTTON_NEUTRAL);
Thread.sleep(100);
getTimerFromFragment().runOneTask();
performClickOnDialog(BUTTON_NEGATIVE);
Thread.sleep(100);
assertThat(getLastCapturedIntent().getAction())
.isEqualTo(ACTION_FLASH_NOTIFICATION_STOP_PREVIEW);
assertStopPreview();
}
@Test
public void clickNeutralAndClickPositive_assertStopPreview() throws InterruptedException {
public void clickNeutralAndClickPositive_assertStopPreview() {
performClickOnDialog(BUTTON_NEUTRAL);
Thread.sleep(100);
getTimerFromFragment().runOneTask();
performClickOnDialog(BUTTON_POSITIVE);
Thread.sleep(100);
assertThat(getLastCapturedIntent().getAction())
.isEqualTo(ACTION_FLASH_NOTIFICATION_STOP_PREVIEW);
assertStopPreview();
}
@Test
public void clickNeutralAndClickColor_assertStartPreview() throws InterruptedException {
public void clickNeutralAndClickColor_assertStartPreview() {
performClickOnDialog(BUTTON_NEUTRAL);
Thread.sleep(100);
getTimerFromFragment().runOneTask();
checkColorButton(CYAN);
Thread.sleep(500);
// When changing the color while the preview is running, the fragment will schedule three
// tasks: stop the current preview, start the new preview, stop the new preview
int numOfPendingTasks = getTimerFromFragment().numOfPendingTasks();
// Run all the pending tasks except the last one
while (numOfPendingTasks > 1) {
getTimerFromFragment().runOneTask();
numOfPendingTasks--;
}
Intent captured = getLastCapturedIntent();
assertThat(captured.getAction()).isEqualTo(ACTION_FLASH_NOTIFICATION_START_PREVIEW);
assertThat(captured.getIntExtra(EXTRA_FLASH_NOTIFICATION_PREVIEW_TYPE, TYPE_SHORT_PREVIEW))
.isEqualTo(TYPE_LONG_PREVIEW);
assertThat(captured.getIntExtra(EXTRA_FLASH_NOTIFICATION_PREVIEW_COLOR, Color.TRANSPARENT))
.isEqualTo(CYAN.mColorInt);
assertStartPreview(CYAN.mColorInt);
}
@Test
@@ -168,6 +170,7 @@ public class ScreenFlashNotificationColorDialogFragmentTest {
checkColorButton(AZURE);
performClickOnDialog(BUTTON_NEGATIVE);
assertThat(getTimerFromFragment()).isNull();
assertThat(mCurrentColor).isEqualTo(ROSE.mColorInt);
}
@@ -193,4 +196,46 @@ public class ScreenFlashNotificationColorDialogFragmentTest {
final int size = capturedIntents.size();
return capturedIntents.get(size - 1);
}
private ScreenFlashNotificationColorDialogFragment createFragment() {
ScreenFlashNotificationColorDialogFragmentWithFakeTimer fragment =
new ScreenFlashNotificationColorDialogFragmentWithFakeTimer();
ReflectionHelpers.setField(fragment, "mCurrentColor", mCurrentColor);
ReflectionHelpers.setField(fragment, "mConsumer",
(Consumer<Integer>) selectedColor -> mCurrentColor = selectedColor);
return fragment;
}
private FakeTimer getTimerFromFragment() {
return (FakeTimer) ReflectionHelpers.getField(mDialogFragment, "mTimer");
}
private void assertStartPreview(int color) {
Intent captured = getLastCapturedIntent();
assertThat(captured.getAction()).isEqualTo(ACTION_FLASH_NOTIFICATION_START_PREVIEW);
assertThat(captured.getIntExtra(EXTRA_FLASH_NOTIFICATION_PREVIEW_TYPE, TYPE_SHORT_PREVIEW))
.isEqualTo(TYPE_LONG_PREVIEW);
assertThat(captured.getIntExtra(EXTRA_FLASH_NOTIFICATION_PREVIEW_COLOR, Color.TRANSPARENT))
.isEqualTo(color);
}
private void assertStopPreview() {
assertThat(getTimerFromFragment().numOfPendingTasks()).isEqualTo(0);
assertThat(getLastCapturedIntent().getAction())
.isEqualTo(ACTION_FLASH_NOTIFICATION_STOP_PREVIEW);
}
/**
* A {@link ScreenFlashNotificationColorDialogFragment} that uses a fake timer so that it won't
* create unmanageable timer threads during test.
*/
public static class ScreenFlashNotificationColorDialogFragmentWithFakeTimer extends
ScreenFlashNotificationColorDialogFragment {
@Override
Timer createTimer() {
return new FakeTimer();
}
}
}

View File

@@ -0,0 +1,81 @@
/*
* Copyright (C) 2023 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.testutils;
import java.util.PriorityQueue;
import java.util.Timer;
import java.util.TimerTask;
/**
* A fake {@link Timer} that doesn't create a TimerThread which is hard to manage in test.
*/
public class FakeTimer extends Timer {
private final PriorityQueue<ScheduledTimerTask> mQueue = new PriorityQueue<>();
public FakeTimer() {
}
@Override
public void cancel() {
mQueue.clear();
}
@Override
public void schedule(TimerTask task, long delay) {
mQueue.offer(new ScheduledTimerTask(System.currentTimeMillis() + delay, task));
}
/**
* Runs the first task in the queue if there's any.
*/
public void runOneTask() {
if (mQueue.size() > 0) {
mQueue.poll().mTask.run();
}
}
/**
* Runs all the queued tasks in order.
*/
public void runAllTasks() {
while (mQueue.size() > 0) {
mQueue.poll().mTask.run();
}
}
/**
* Returns number of pending tasks in the timer
*/
public int numOfPendingTasks() {
return mQueue.size();
}
private static class ScheduledTimerTask implements Comparable<ScheduledTimerTask> {
final long mTimeToRunInMillisSeconds;
final TimerTask mTask;
ScheduledTimerTask(long timeToRunInMilliSeconds, TimerTask task) {
this.mTimeToRunInMillisSeconds = timeToRunInMilliSeconds;
this.mTask = task;
}
@Override
public int compareTo(ScheduledTimerTask other) {
return Long.compare(this.mTimeToRunInMillisSeconds, other.mTimeToRunInMillisSeconds);
}
}
}