Async Health Indicator for spring-boot-actuator >=2.2 and >=3.0 gives Health Indicator the ability to get refreshed asynchronously on a background ThreadPoolExecutor using the annotation @AsyncHealth
.
The health()
method is invoked on application startup and with the configured refreshRate
between the termination of one execution and the commencement of the next.
Calling the /health
endpoint will not invoke the health()
method but return the last Health
calculated asynchronously. (Status.UNKNOWN
is returned if the /health
endpoint is called before the first HealthIndicator.health()
check is completed (likely to occur on application startup))
The duration of the health()
method is monitored. If it exceeds the configured timeout
, any call to /health
will return this HealthIndicator as DOWN
until the next health()
method completion.
The following details are added to the pre-existing Health
details:
lastChecked
(LocalDateTime
): Defines when that lastHealth
started execution.lastDuration
(Duration in ms): Defines how long it took to calculate that lastHealth
.reason
(enum): Added only if the Health Indicator is markedDOWN
:Exception
orTimeout
.
- Expensive Health Indicators that do not have to be calculated every time
/health
is called can run on their own schedule. - The call to
/health
is instantaneous as it returns pre-calculatedHealth
. - Multiple Health Indicators can run in parallel.
- Timeouts can be controlled per Health Indicator.
This module is auto-configured.
- Include dependency from maven central:
For spring boot 2:
<dependency>
<groupId>com.teketik</groupId>
<artifactId>async-health-indicator</artifactId>
<version>boot2-v1.3</version>
</dependency>
For spring boot 3:
<dependency>
<groupId>com.teketik</groupId>
<artifactId>async-health-indicator</artifactId>
<version>boot3-v1.4</version>
</dependency>
- Annotate any
HealthIndicator
with@AsyncHealth(refreshRate = $REFRESH_RATE, timeout = $TIMEOUT, interruptOnTimeout = $INTERRUPT_ON_TIMEOUT)
$REFRESH_RATE
= Fixed delay in seconds between the termination of the health()
execution and the commencement of the next (default 1
).
$TIMEOUT
= The maximum time in seconds that the health()
execution can take before being considered DOWN
(default 10
).
$INTERRUPT_ON_TIMEOUT
= Whether the thread should be interrupted when a timeout occurs. (if false
, the next check will only be scheduled after the current execution terminates naturally). (default true
)
When a health()
method duration exceeds the configured timeout
, the thread running it is interrupted
if interruptOnTimeout
is true
with the hope that the method will fail with an exception (causing it to be DOWN
) and free up the thread.
Unfortunately, most I/O calls are not interruptible and the thread may continue to execute the method until it times out (according to the libraries and configuration used).
If that happens, you will observe the timeout
error message printed for each /health
hit until that method times out like:
ERROR AsyncHealthIndicator : HealthIndicator[name=myIndicator] is taking too long to execute [duration=2121ms][timeout=2s]
ERROR AsyncHealthIndicator : HealthIndicator[name=myIndicator] is taking too long to execute [duration=3189ms][timeout=2s]
It is therefore recommended to ensure that your health()
methods can time out naturally within an acceptable window (matching the configured timeout
)
Property | Description | Default |
---|---|---|
management.health.async.pool.max-size |
Number of maximum threads calculating the health() methods. (Note that this max will likely be reached on application startup when all indicators are starting up but will likely size down when different durations allow threads to be reused more efficiently). |
10 |
management.health.async.pool.keep-alive |
Maximum time that excess idle threads will wait for new tasks before terminating. | 10 |
- On application startup, all cached health indicators are logged under logger
com.teketik.spring.health.AsyncHealthIndicatorAutoConfiguration
asINFO
:
Example: Initializing AsyncHealthIndicator[name=myIndicator][refreshRate=3s][timeout=2s]
- All
@AsyncHealth
annotatedHealthIndicator
s have details logged under loggercom.teketik.spring.health.AsyncHealthIndicator
asDEBUG
:
Example: HealthIndicator[name=myIndicator][duration=147ms][status=UP {detailKey=detailValue}]
Only implementations of HealthIndicator
are currently supported. Composites
are not. (Let me know if you need that!).