ratpack.health.HealthCheckHandler Maven / Gradle / Ivy
/*
* Copyright 2015 the original author or authors.
*
* 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 ratpack.health;
import com.google.common.collect.ImmutableSortedMap;
import com.google.common.reflect.TypeToken;
import ratpack.exec.Execution;
import ratpack.exec.Promise;
import ratpack.exec.Throttle;
import ratpack.handling.Context;
import ratpack.handling.Handler;
import ratpack.registry.Registry;
import ratpack.util.Types;
import java.util.Collections;
import java.util.Iterator;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;
/**
* A handler that executes {@link HealthCheck health checks} and renders the results.
*
* The handler obtains health checks via the context's registry.
* Typically, health checks are added to the server registry.
*
{@code
* import ratpack.exec.Promise;
* import ratpack.registry.Registry;
* import ratpack.health.HealthCheck;
* import ratpack.health.HealthCheckHandler;
* import ratpack.test.embed.EmbeddedApp;
*
* import static org.junit.Assert.*;
*
* public class Example {
*
* static class ExampleHealthCheck implements HealthCheck {
* public String getName() {
* return "example"; // must be unique within the application
* }
*
* public Promise check(Registry registry) throws Exception {
* return Promise.value(HealthCheck.Result.healthy());
* }
* }
*
* public static void main(String... args) throws Exception {
* EmbeddedApp.of(s -> s
* .registryOf(r -> r
* .add(new ExampleHealthCheck())
* .add(HealthCheck.of("inline", registry -> // alternative way to implement health checks
* Promise.value(HealthCheck.Result.unhealthy("FAILED"))
* ))
* )
* .handlers(c -> c
* .get("health/:name?", new HealthCheckHandler())
* )
* ).test(httpClient -> {
* // Run all health checks
* String result = httpClient.getText("health");
* String[] results = result.split("\n");
* assertEquals(2, results.length);
* assertEquals("example : HEALTHY", results[0]);
* assertEquals("inline : UNHEALTHY [FAILED]", results[1]);
*
* // Run a single health check
* result = httpClient.getText("health/example");
* assertEquals("example : HEALTHY", result);
* });
* }
* }
* }
*
* The output format
*
* The handler creates a {@link HealthCheckResults} object with the results of running the health checks and {@link Context#render(Object) renders} it.
* Ratpack provides a default renderer for {@link HealthCheckResults} objects, that renders results as plain text one per line with the format:
*
{@code name : HEALTHY|UNHEALTHY [message] [exception]}
* If any result is unhealthy, a {@code 503} status will be emitted, else {@code 200}.
*
* To change the output format, simply add your own renderer for this type to the registry.
*
*
Concurrency
*
* The concurrency of the health check execution is managed by a {@link Throttle}.
* By default, an {@link Throttle#unlimited() unlimited throttle} is used.
* A sized throttle can be explicitly given to the constructor.
*
*
Rendering single health checks
*
* The handler checks for the presence of a {@link Context#getPathTokens() path token} to indicate the name of an individual check to execute.
* By default, the token is named {@value #DEFAULT_NAME_TOKEN}.
* If a path token is present, but indicates the name of a non-existent health check, a {@code 404} {@link Context#clientError(int) client error} will be raised.
*
* @see ratpack.health.HealthCheck
* @see ratpack.health.HealthCheckResults
*/
public class HealthCheckHandler implements Handler {
/**
* The default path token name that indicates the individual health check to run.
*
* Value: {@value}
*
* @see #HealthCheckHandler(String, Throttle)
*/
public static final String DEFAULT_NAME_TOKEN = "name";
private static final TypeToken HEALTH_CHECK_TYPE_TOKEN = Types.token(HealthCheck.class);
private final String name;
private final Throttle throttle;
/**
* Uses the default values of {@link #DEFAULT_NAME_TOKEN} and an unlimited throttle.
*
* @see #HealthCheckHandler(String, Throttle)
*/
public HealthCheckHandler() {
this(DEFAULT_NAME_TOKEN, Throttle.unlimited());
}
/**
* Uses an unlimited throttle and the given name for the health check identifying path token.
*
* @param pathTokenName health check name
* @see #HealthCheckHandler(String, Throttle)
*/
public HealthCheckHandler(String pathTokenName) {
this(pathTokenName, Throttle.unlimited());
}
/**
* Uses the {@link #DEFAULT_NAME_TOKEN} and the given throttle.
*
* @param throttle the throttle for health check execution
* @see #HealthCheckHandler(String, Throttle)
*/
public HealthCheckHandler(Throttle throttle) {
this(DEFAULT_NAME_TOKEN, throttle);
}
/**
* Constructor.
*
* The path token name parameter specifies the name of the path token that, if present, indicates the single health check to execute.
* If this path token is not present, all checks will be run.
*
* The throttle controls the concurrency of health check execution.
* The actual parallelism of the health check executions is ultimately determined by the size of the application event loop in conjunction with the throttle size.
* Generally, an {@link Throttle#unlimited() unlimited throttle} is appropriate unless there are many health checks that are executed frequently as this may degrade the performance of other requests.
* To serialize health check execution, use a throttle of size 1.
*
* @param pathTokenName the name of health check
* @param throttle the throttle for health check execution
*/
protected HealthCheckHandler(String pathTokenName, Throttle throttle) {
this.name = pathTokenName;
this.throttle = throttle;
}
/**
* The throttle for executing health checks.
*
* @return the throttle for executing health checks.
*/
public Throttle getThrottle() {
return throttle;
}
/**
* The name of the path token that may indicate a particular health check to execute.
*
* @return the name of the path token that may indicate a particular health check to execute
*/
public String getName() {
return name;
}
/**
* Renders health checks.
*
* @param ctx the request context
*/
@Override
public void handle(Context ctx) {
ctx.getResponse().getHeaders()
.add("Cache-Control", "no-cache, no-store, must-revalidate")
.add("Pragma", "no-cache")
.add("Expires", 0);
try {
String checkName = ctx.getPathTokens().get(name);
if (checkName != null) {
Optional first = ctx.first(HEALTH_CHECK_TYPE_TOKEN, healthCheck -> healthCheck.getName().equals(checkName) ? healthCheck : null);
if (first.isPresent()) {
ctx.render(execute(ctx, Collections.singleton(first.get())));
} else {
ctx.clientError(404);
}
} else {
ctx.render(execute(ctx, ctx.getAll(HEALTH_CHECK_TYPE_TOKEN)));
}
} catch (Exception e) {
ctx.error(e);
}
}
private Promise execute(Registry registry, HealthCheck healthCheck) {
return Promise.wrap(() -> healthCheck.check(registry)).mapError(HealthCheck.Result::unhealthy);
}
private Promise execute(Registry registry, Iterable extends HealthCheck> healthChecks) {
Iterator extends HealthCheck> iterator = healthChecks.iterator();
if (!iterator.hasNext()) {
return Promise.value(new HealthCheckResults(ImmutableSortedMap.of()));
}
return Promise.