Files
java-design-patterns/health-check/src/test/java/DatabaseTransactionHealthIndicatorTest.java
T
Doksanbir 21f7b026f5 pattern: Implement Health Check for Microservices Observability (#2695) (#2774)
* Add Health Check pattern implementation

The commit introduces  Health Check pattern, providing a series of health indicators for system performance and stability monitoring, including checks for system CPU load, process CPU load, database health, memory usage, and garbage collection metrics. It also includes asynchronous execution and caching mechanisms for health checks, and retry configurations for resilience.

Implements health checking components as per issue #2695.

* Test cases and javadoc for HealthEndpointIntegrationTest

* Added more log to test case to see why it returns 503

* Change config values to see if the system High system CPU load is resolved or not in CI.

* Fixes for test cases.

* some fixes for Sonar.

* some fixes for Sonar.
ADDED HIGH_PROCESS_CPU_LOAD_MESSAGE_WITHOUT_PARAM
ADDED HIGH_SYSTEM_CPU_LOAD_MESSAGE_WITHOUT_PARAM

* Sonar fixes address "Define and throw a dedicated exception instead of using a generic one."

added HealthCheckInterruptedException
refactored CustomHealthIndicator

* fixes checkstyle violation.
2023-12-02 14:17:01 +02:00

119 lines
4.3 KiB
Java

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.*;
import com.iluwatar.health.check.AsynchronousHealthChecker;
import com.iluwatar.health.check.DatabaseTransactionHealthIndicator;
import com.iluwatar.health.check.HealthCheckRepository;
import java.util.concurrent.CompletableFuture;
import java.util.function.Supplier;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import org.springframework.boot.actuate.health.Health;
import org.springframework.boot.actuate.health.Status;
import org.springframework.retry.support.RetryTemplate;
/**
* Unit tests for the {@link DatabaseTransactionHealthIndicator} class.
*
* @author ydoksanbir
*/
class DatabaseTransactionHealthIndicatorTest {
/** Timeout value in seconds for the health check. */
private final long timeoutInSeconds = 4;
/** Mocked HealthCheckRepository instance. */
@Mock private HealthCheckRepository healthCheckRepository;
/** Mocked AsynchronousHealthChecker instance. */
@Mock private AsynchronousHealthChecker asynchronousHealthChecker;
/** Mocked RetryTemplate instance. */
@Mock private RetryTemplate retryTemplate;
/** `DatabaseTransactionHealthIndicator` instance to be tested. */
private DatabaseTransactionHealthIndicator healthIndicator;
/** Performs initialization before each test method. */
@BeforeEach
void setUp() {
MockitoAnnotations.openMocks(this);
healthIndicator =
new DatabaseTransactionHealthIndicator(
healthCheckRepository, asynchronousHealthChecker, retryTemplate);
healthIndicator.setTimeoutInSeconds(timeoutInSeconds);
}
/**
* Test case for the `health()` method when the database transaction succeeds.
*
* <p>Asserts that when the `health()` method is called and the database transaction succeeds, it
* returns a Health object with Status.UP.
*/
@Test
void whenDatabaseTransactionSucceeds_thenHealthIsUp() {
CompletableFuture<Health> future = CompletableFuture.completedFuture(Health.up().build());
when(asynchronousHealthChecker.performCheck(any(Supplier.class), eq(timeoutInSeconds)))
.thenReturn(future);
// Simulate the health check repository behavior
doNothing().when(healthCheckRepository).performTestTransaction();
// Now call the actual method
Health health = healthIndicator.health();
// Check that the health status is UP
assertEquals(Status.UP, health.getStatus());
}
/**
* Test case for the `health()` method when the database transaction fails.
*
* <p>Asserts that when the `health()` method is called and the database transaction fails, it
* returns a Health object with Status.DOWN.
*/
@Test
void whenDatabaseTransactionFails_thenHealthIsDown() {
CompletableFuture<Health> future = new CompletableFuture<>();
when(asynchronousHealthChecker.performCheck(any(Supplier.class), eq(timeoutInSeconds)))
.thenReturn(future);
// Simulate a database exception during the transaction
doThrow(new RuntimeException("DB exception"))
.when(healthCheckRepository)
.performTestTransaction();
// Complete the future exceptionally to simulate a failure in the health check
future.completeExceptionally(new RuntimeException("DB exception"));
Health health = healthIndicator.health();
// Check that the health status is DOWN
assertEquals(Status.DOWN, health.getStatus());
}
/**
* Test case for the `health()` method when the health check times out.
*
* <p>Asserts that when the `health()` method is called and the health check times out, it returns
* a Health object with Status.DOWN.
*/
@Test
void whenHealthCheckTimesOut_thenHealthIsDown() {
CompletableFuture<Health> future = new CompletableFuture<>();
when(asynchronousHealthChecker.performCheck(any(Supplier.class), eq(timeoutInSeconds)))
.thenReturn(future);
// Complete the future exceptionally to simulate a timeout
future.completeExceptionally(new RuntimeException("Simulated timeout"));
Health health = healthIndicator.health();
// Check that the health status is DOWN due to timeout
assertEquals(Status.DOWN, health.getStatus());
}
}