/*
* Copyright (C) 2020 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.car.test.mocks;
import android.annotation.NonNull;
import android.annotation.Nullable;
import android.car.test.util.Visitor;
import android.util.Log;
import com.android.car.internal.util.FunctionalUtils;
import org.mockito.invocation.InvocationOnMock;
import org.mockito.stubbing.Answer;
import java.util.concurrent.CountDownLatch;
/**
* Custom Mockito {@link Answer} that blocks until a condition is unblocked by the test case.
*
*
Example:
*
*
{@code
*
* BlockingAnswer{@code } blockingAnswer = BlockingAnswer.forVoidReturn(
* 10_000, (invocation) -> {
* MyObject obj = (MyObject) invocation.getArguments()[0];
* obj.doSomething();
* });
* doAnswer(blockingAnswer).when(mMock).mockSomething();
* doSomethingOnTest();
* blockingAnswer.unblock();
*
* }
*/
public final class BlockingAnswer implements Answer {
private static final String TAG = BlockingAnswer.class.getSimpleName();
private final CountDownLatch mLatch = new CountDownLatch(1);
private final long mTimeoutMs;
@NonNull
private final Visitor mInvocator;
@Nullable
private final T mValue;
private BlockingAnswer(long timeoutMs, @NonNull Visitor invocator,
@Nullable T value) {
mTimeoutMs = timeoutMs;
mInvocator = invocator;
mValue = value;
}
/**
* Unblocks the answer so the test case can continue.
*/
public void unblock() {
Log.v(TAG, "Unblocking " + mLatch);
mLatch.countDown();
}
/**
* Creates a {@link BlockingAnswer} for a {@code void} method.
*
* @param timeoutMs maximum time the answer would block.
*
* @param invocator code executed after the answer is unblocked.
*/
public static BlockingAnswer forVoidReturn(long timeoutMs,
@NonNull Visitor invocator) {
return new BlockingAnswer(timeoutMs, invocator, /* value= */ null);
}
/**
* Creates a {@link BlockingAnswer} for a method that returns type {@code T}
*
* @param timeoutMs maximum time the answer would block.
* @param invocator code executed after the answer is unblocked.
*
* @return what the answer should return
*/
public static BlockingAnswer forReturn(long timeoutMs,
@NonNull Visitor invocator, @Nullable T value) {
return new BlockingAnswer(timeoutMs, invocator, value);
}
@Override
public T answer(InvocationOnMock invocation) throws Throwable {
String threadName = "BlockingAnswerThread";
Log.d(TAG, "Starting thread " + threadName);
new Thread(() -> {
JavaMockitoHelper.silentAwait(mLatch, mTimeoutMs);
Log.d(TAG, "Invocating " + FunctionalUtils.getLambdaName(mInvocator));
mInvocator.visit(invocation);
}, threadName).start();
Log.d(TAG, "Returning " + mValue);
return mValue;
}
}