blob: 367f70a6ca830b7f8a5f139d02861d320f1ec955 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The SF licenses this file
* to you 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 org.apache.sling.hc.core.impl.executor;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertTrue;
import static org.mockito.Mockito.times;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
import static org.mockito.MockitoAnnotations.initMocks;
import java.util.Collection;
import java.util.Date;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;
import java.util.TreeSet;
import org.apache.sling.hc.api.Result;
import org.apache.sling.hc.api.Result.Status;
import org.apache.sling.hc.api.ResultLog.Entry;
import org.apache.sling.hc.api.execution.HealthCheckExecutionResult;
import org.apache.sling.hc.util.HealthCheckMetadata;
import org.junit.Before;
import org.junit.Test;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.mockito.Spy;
public class HealthCheckExecutorImplTest {
@InjectMocks
private HealthCheckExecutorImpl healthCheckExecutorImpl = new HealthCheckExecutorImpl();;
@Mock
private HealthCheckFuture future;
@Mock
private HealthCheckMetadata HealthCheckMetadata;
@Spy
private HealthCheckResultCache healthCheckResultCache = new HealthCheckResultCache();
@Before
public void setup() {
initMocks(this);
when(future.getHealthCheckMetadata()).thenReturn(HealthCheckMetadata);
when(HealthCheckMetadata.getTitle()).thenReturn("Test Check");
// 2 sec normal timeout
healthCheckExecutorImpl.setTimeoutInMs(2000L);
// 10 sec timeout for critical
healthCheckExecutorImpl.setLongRunningFutureThresholdForRedMs(10000L);
}
@Test
public void testCollectResultsFromFutures() throws Exception {
List<HealthCheckFuture> futures = new LinkedList<HealthCheckFuture>();
futures.add(future);
Collection<HealthCheckExecutionResult> results = new TreeSet<HealthCheckExecutionResult>();
when(future.isDone()).thenReturn(true);
ExecutionResult testResult = new ExecutionResult(HealthCheckMetadata, new Result(Result.Status.OK, "test"), 10L);
when(future.get()).thenReturn(testResult);
healthCheckExecutorImpl.collectResultsFromFutures(futures, results);
verify(future, times(1)).get();
assertEquals(1, results.size());
assertTrue(results.contains(testResult));
}
@Test
public void testCollectResultsFromFuturesTimeout() throws Exception {
// add an earlier result with status ok (that will be shown as part of the log)
addResultToCache(Status.OK);
List<HealthCheckFuture> futures = new LinkedList<HealthCheckFuture>();
futures.add(future);
Set<HealthCheckExecutionResult> results = new TreeSet<HealthCheckExecutionResult>();
when(future.isDone()).thenReturn(false);
// simulating a future that was created 5sec ago
when(future.getCreatedTime()).thenReturn(new Date(new Date().getTime() - 1000 * 5));
healthCheckExecutorImpl.collectResultsFromFutures(futures, results);
verify(future, times(0)).get();
assertEquals(1, results.size());
HealthCheckExecutionResult result = results.iterator().next();
assertEquals(Result.Status.WARN, result.getHealthCheckResult().getStatus());
// 3 because previous result exists and is part of log
assertEquals(3, getLogEntryCount(result));
}
@Test
public void testCollectResultsFromFuturesCriticalTimeout() throws Exception {
List<HealthCheckFuture> futures = new LinkedList<HealthCheckFuture>();
futures.add(future);
Set<HealthCheckExecutionResult> results = new TreeSet<HealthCheckExecutionResult>();
when(future.isDone()).thenReturn(false);
// use an old date now (simulating a future that has run for an hour)
when(future.getCreatedTime()).thenReturn(new Date(new Date().getTime() - 1000 * 60 * 60));
healthCheckExecutorImpl.collectResultsFromFutures(futures, results);
assertEquals(1, results.size());
HealthCheckExecutionResult result = results.iterator().next();
verify(future, times(0)).get();
assertEquals(Result.Status.CRITICAL, result.getHealthCheckResult().getStatus());
assertEquals(1, getLogEntryCount(result));
}
@Test
public void testCollectResultsFromFuturesWarnTimeoutWithPreviousCritical() throws Exception {
// an earlier result with critical
addResultToCache(Status.CRITICAL);
List<HealthCheckFuture> futures = new LinkedList<HealthCheckFuture>();
futures.add(future);
Set<HealthCheckExecutionResult> results = new TreeSet<HealthCheckExecutionResult>();
when(future.isDone()).thenReturn(false);
// simulating a future that was created 5sec ago
when(future.getCreatedTime()).thenReturn(new Date(new Date().getTime() - 1000 * 5));
healthCheckExecutorImpl.collectResultsFromFutures(futures, results);
assertEquals(1, results.size());
HealthCheckExecutionResult result = results.iterator().next();
verify(future, times(0)).get();
// expect CRITICAL because previous result (before timeout) was CRITICAL (and not only WARN)
assertEquals(Result.Status.CRITICAL, result.getHealthCheckResult().getStatus());
assertEquals(3, getLogEntryCount(result));
}
private int getLogEntryCount(HealthCheckExecutionResult result) {
int logEntryCount = 0;
final Iterator<Entry> it = result.getHealthCheckResult().iterator();
while(it.hasNext()) {
it.next();
logEntryCount++;
}
return logEntryCount;
}
private void addResultToCache(Status status) {
healthCheckResultCache.updateWith(new ExecutionResult(HealthCheckMetadata, new Result(status, "Status "+status), 1000));
}
}