| /* |
| * 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 com.android.compatibility.common.util; |
| |
| import static junit.framework.Assert.fail; |
| |
| import android.os.SystemClock; |
| import android.util.Log; |
| |
| import java.util.function.BooleanSupplier; |
| |
| public class TestUtils { |
| private static final String TAG = "CtsTestUtils"; |
| |
| private TestUtils() { |
| } |
| |
| public static final int DEFAULT_TIMEOUT_SECONDS = 30; |
| |
| /** Print an error log and fail. */ |
| public static void failWithLog(String message) { |
| Log.e(TAG, message); |
| fail(message); |
| } |
| |
| @FunctionalInterface |
| public interface BooleanSupplierWithThrow { |
| boolean getAsBoolean() throws Exception; |
| } |
| |
| @FunctionalInterface |
| public interface RunnableWithThrow { |
| void run() throws Exception; |
| } |
| |
| /** |
| * Wait until {@code predicate} is satisfied, or fail, with {@link #DEFAULT_TIMEOUT_SECONDS}. |
| */ |
| public static void waitUntil(String message, BooleanSupplierWithThrow predicate) |
| throws Exception { |
| waitUntil(message, 0, predicate); |
| } |
| |
| /** |
| * Wait until {@code predicate} is satisfied, or fail, with a given timeout. |
| */ |
| public static void waitUntil( |
| String message, int timeoutSecond, BooleanSupplierWithThrow predicate) |
| throws Exception { |
| if (timeoutSecond <= 0) { |
| timeoutSecond = DEFAULT_TIMEOUT_SECONDS; |
| } |
| int sleep = 125; |
| final long timeout = SystemClock.uptimeMillis() + timeoutSecond * 1000; |
| while (SystemClock.uptimeMillis() < timeout) { |
| if (predicate.getAsBoolean()) { |
| return; // okay |
| } |
| Thread.sleep(sleep); |
| sleep *= 5; |
| sleep = Math.min(2000, sleep); |
| } |
| failWithLog("Timeout: " + message); |
| } |
| |
| /** |
| * Run a Runnable {@code r}, and if it throws, also run {@code onFailure}. |
| */ |
| public static void runWithFailureHook(RunnableWithThrow r, RunnableWithThrow onFailure) |
| throws Exception { |
| if (r == null) { |
| throw new NullPointerException("r"); |
| } |
| if (onFailure == null) { |
| throw new NullPointerException("onFailure"); |
| } |
| try { |
| r.run(); |
| } catch (Throwable th) { |
| Log.e(TAG, "Caught exception: " + th, th); |
| onFailure.run(); |
| throw th; |
| } |
| } |
| |
| /** |
| * Synchronized wait for a specified condition. |
| * |
| * @param notifyLock Lock that will be notified when the condition might have changed |
| * @param condition The condition to check |
| * @param timeoutMs The timeout in millis |
| * @param conditionName The name to include in the assertion. If null, will be given a default. |
| */ |
| public static void waitOn(Object notifyLock, BooleanSupplier condition, |
| long timeoutMs, String conditionName) { |
| if (conditionName == null) conditionName = "condition"; |
| if (condition.getAsBoolean()) return; |
| |
| synchronized (notifyLock) { |
| try { |
| long timeSlept = 0; |
| while (!condition.getAsBoolean() && timeSlept < timeoutMs) { |
| long sleepStart = SystemClock.uptimeMillis(); |
| notifyLock.wait(timeoutMs - timeSlept); |
| timeSlept += SystemClock.uptimeMillis() - sleepStart; |
| } |
| if (!condition.getAsBoolean()) { |
| throw new AssertionError("Timed out after " + timeSlept |
| + "ms waiting for: " + conditionName); |
| } |
| } catch (InterruptedException e) { |
| throw new RuntimeException(e); |
| } |
| } |
| } |
| } |
| |