com.dasasian.chok.testutil.TestUtil Maven / Gradle / Ivy
/**
* Copyright (C) 2014 Dasasian ([email protected])
*
* 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.dasasian.chok.testutil;
import com.dasasian.chok.client.Client;
import com.dasasian.chok.master.Master;
import com.dasasian.chok.node.Node;
import com.dasasian.chok.protocol.InteractionProtocol;
import com.dasasian.chok.protocol.metadata.IndexMetaData;
import com.dasasian.chok.protocol.metadata.IndexMetaData.Shard;
import com.dasasian.chok.util.ZkConfiguration;
import com.dasasian.chok.util.ZkConfiguration.PathDef;
import org.I0Itec.zkclient.exception.ZkNoNodeException;
import org.mockito.Mockito;
import org.mockito.exceptions.base.MockitoAssertionError;
import org.mockito.stubbing.Stubber;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.List;
import java.util.Set;
import java.util.concurrent.Callable;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import static org.fest.assertions.Assertions.assertThat;
import static org.junit.Assert.assertEquals;
public class TestUtil {
private static final Logger LOG = LoggerFactory.getLogger(TestUtil.class);
/**
* This waits until the provided {@link Callable} returns an object that is
* equals to the given expected value or the timeout has been reached. In both
* cases this method will return the return value of the latest
* {@link Callable} execution.
*
* @param expectedValue The expected value of the callable.
* @param callable The callable.
* @param The return type of the callable.
* @param timeUnit The timeout timeunit.
* @param timeout The timeout.
* @return the return value of the latest {@link Callable} execution.
* @throws Exception when an error occurs
* @throws InterruptedException when interrupted
*/
public static T waitUntil(T expectedValue, Callable callable, TimeUnit timeUnit, long timeout) throws Exception {
long startTime = System.currentTimeMillis();
do {
T actual = callable.call();
if (expectedValue.equals(actual)) {
return actual;
}
if (System.currentTimeMillis() > startTime + timeUnit.toMillis(timeout)) {
return actual;
}
Thread.sleep(50);
} while (true);
}
/**
* This waits until a mockito verification passed (which is provided in the
* runnable). This waits until the virification passed or the timeout has been
* reached. If the timeout has been reached this method will rethrow the
* {@link MockitoAssertionError} that comes from the mockito verification
* code.
*
* @param runnable The runnable containing the mockito verification.
* @param timeUnit The timeout timeunit.
* @param timeout The timeout.
* @throws InterruptedException when interrupted
*/
public static void waitUntilVerified(Runnable runnable, TimeUnit timeUnit, int timeout) throws InterruptedException {
LOG.debug("Waiting for " + timeout + " " + timeUnit + " until verified.");
long startTime = System.currentTimeMillis();
do {
MockitoAssertionError exception = null;
try {
runnable.run();
} catch (MockitoAssertionError e) {
exception = e;
}
if (exception == null) {
return;
}
if (System.currentTimeMillis() > startTime + timeUnit.toMillis(timeout)) {
LOG.debug("Timeout reached without satifying expectations.");
throw exception;
}
Thread.sleep(50);
} while (true);
}
public static void waitUntilNoExceptionThrown(Runnable runnable, TimeUnit timeUnit, int timeout) throws InterruptedException {
long startTime = System.currentTimeMillis();
do {
RuntimeException exception = null;
try {
runnable.run();
} catch (RuntimeException e) {
exception = e;
}
if (exception == null) {
return;
}
if (System.currentTimeMillis() > startTime + timeUnit.toMillis(timeout)) {
throw exception;
}
Thread.sleep(50);
} while (true);
}
/**
* Creates a Mockito answer object that can be used for asynchronously
* stubbing. For example:
*
*
* final CountDownLatch countDownLatch = new CountDownLatch(1);
* Mockito.doAnswer(TestUtil.createCountDownAnswer(countDownLatch)).when(listener).announceNode(nodeName);
* mock.someMethod(someValue);
* countDownLatch.await(10, TimeUnit.SECONDS);
* Assert.assertEquals("expecting invocation within 10 seconds", 0, countDownLatch.getCount());
*
* @param countDownLatch the countdown latch
* @return the stubber
*/
public static Stubber createCountDownAnswer(final CountDownLatch countDownLatch) {
return Mockito.doAnswer(invocation -> {
countDownLatch.countDown();
return null;
});
}
public static void waitUntilLeaveSafeMode(final Master master) throws Exception {
waitUntil(false, master::isInSafeMode, TimeUnit.SECONDS, 30);
assertEquals(false, master.isInSafeMode());
}
public static void waitUntilBecomeMaster(final Master master) throws Exception {
waitUntil(false, () -> !master.isMaster(), TimeUnit.SECONDS, 30);
assertEquals(true, master.isMaster());
}
public static void waitUntilIndexDeployed(final InteractionProtocol protocol, final String indexName) throws Exception {
waitUntil(false, () -> protocol.getIndexMD(indexName) == null, TimeUnit.SECONDS, 30);
}
public static void waitUntilIndexBalanced(final InteractionProtocol protocol, final String indexName) throws Exception {
waitUntil(true, () -> {
IndexMetaData indexMD = protocol.getIndexMD(indexName);
if (indexMD.hasDeployError()) {
throw new IllegalStateException("index " + indexName + " has a deploy error");
}
return protocol.getReplicationReport(indexMD).isBalanced();
}, TimeUnit.SECONDS, 30);
}
public static void waitUntilShardsUndeployed(final InteractionProtocol protocol, final IndexMetaData indexMD) throws Exception {
TestUtil.waitUntil(false, () -> {
int nodeCount = 0;
Set shards = indexMD.getShards();
for (Shard shard : shards) {
try {
nodeCount += protocol.getShardNodes(shard.getName()).size();
} catch (ZkNoNodeException e) {
// deleted already
}
}
return nodeCount != 0;
}, TimeUnit.SECONDS, 30);
}
public static void waitUntilNodeServesShards(final InteractionProtocol protocol, final String nodeName, final int shardCount) throws Exception {
waitUntil(true, () -> protocol.getNodeShards(nodeName).size() == shardCount, TimeUnit.SECONDS, 30);
assertEquals(shardCount, protocol.getNodeShards(nodeName).size());
}
public static void waitUntilNumberOfLiveNode(final InteractionProtocol protocol, final int nodeCount) throws Exception {
waitUntil(true, () -> protocol.getLiveNodes().size() == nodeCount, TimeUnit.SECONDS, 30);
assertEquals(nodeCount, protocol.getLiveNodes().size());
}
public static void waitUntilEmptyOperationQueues(final InteractionProtocol protocol, final Master master, final List nodes) throws Exception {
waitUntil(true, () -> operationQueuesEmpty(protocol, master, nodes), TimeUnit.SECONDS, 30);
assertThat(operationQueuesEmpty(protocol, master, nodes)).as("node operation queues are empty").isTrue();
}
private static Boolean operationQueuesEmpty(final InteractionProtocol protocol, Master master, final List nodes) {
ZkConfiguration zkConf = protocol.getZkConfiguration();
if (protocol.getZkClient().countChildren(zkConf.getPath(PathDef.MASTER_QUEUE, "operations")) > 0) {
return false;
}
for (Node node : nodes) {
if (protocol.getZkClient().countChildren(zkConf.getPath(PathDef.NODE_QUEUE, node.getName(), "operations")) > 0) {
return false;
}
}
return true;
}
public static void waitUntilClientHasIndex(Client client, String indexName) throws InterruptedException {
while (!client.hasIndex(indexName)) {
Thread.sleep(10);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy