package io.vertx.ext.healthchecks.impl;
import io.vertx.core.AsyncResult;
import io.vertx.core.Handler;
import io.vertx.core.Promise;
import io.vertx.core.Vertx;
import io.vertx.core.http.HttpHeaders;
import io.vertx.core.http.HttpMethod;
import io.vertx.core.http.HttpServerResponse;
import io.vertx.core.impl.logging.Logger;
import io.vertx.core.impl.logging.LoggerFactory;
import io.vertx.core.json.JsonObject;
import io.vertx.ext.auth.authentication.AuthenticationProvider;
import io.vertx.ext.healthchecks.HealthCheckHandler;
import io.vertx.ext.healthchecks.HealthChecks;
import io.vertx.ext.healthchecks.CheckResult;
import io.vertx.ext.healthchecks.Status;
import io.vertx.ext.web.RoutingContext;
import java.util.List;
import java.util.Objects;
import static io.vertx.ext.healthchecks.CheckResult.isUp;
public class HealthCheckHandlerImpl implements HealthCheckHandler {
private Logger log = LoggerFactory.getLogger(HealthCheckHandler.class);
private HealthChecks healthChecks;
private final AuthenticationProvider authProvider;
public HealthCheckHandlerImpl(Vertx vertx, AuthenticationProvider provider) {
this.healthChecks = new HealthChecksImpl(vertx);
this.authProvider = provider;
}
public HealthCheckHandlerImpl(HealthChecks hc, AuthenticationProvider provider) {
this.healthChecks = Objects.requireNonNull(hc);
this.authProvider = provider;
}
@Override
public HealthCheckHandler register(String name, Handler<Promise<Status>> procedure) {
healthChecks.register(name, procedure);
return this;
}
@Override
public HealthCheckHandler register(String name, long timeout, Handler<Promise<Status>> procedure) {
healthChecks.register(name, timeout, procedure);
return this;
}
@Override
public void handle(RoutingContext rc) {
String path = rc.request().path();
String mount = rc.mountPoint();
String route = rc.currentRoute().getPath();
String id;
if (mount != null && path.startsWith(mount)) {
path = path.substring(mount.length());
}
if (route != null && path.startsWith(route)) {
id = path.substring(route.length());
} else {
id = path;
}
if (authProvider != null) {
JsonObject authData = new JsonObject();
rc.request().headers().forEach(entry -> authData.put(entry.getKey(), entry.getValue()));
rc.request().params().forEach(entry -> authData.put(entry.getKey(), entry.getValue()));
if (rc.request().method() == HttpMethod.POST
&& rc.request().getHeader(HttpHeaders.CONTENT_TYPE) != null
&& rc.request().getHeader(HttpHeaders.CONTENT_TYPE).contains("application/json")) {
try {
JsonObject body = rc.getBodyAsJson();
if (body != null) {
authData.mergeIn(body);
}
} catch (Exception err) {
log.error("Invalid authentication json body", err);
}
}
authProvider.authenticate(authData, ar -> {
if (ar.failed()) {
rc.response().setStatusCode(403).end();
} else {
healthChecks.checkStatus(id, healthReportHandler(rc));
}
});
} else {
healthChecks.checkStatus(id, healthReportHandler(rc));
}
}
private Handler<AsyncResult<CheckResult>> healthReportHandler(RoutingContext rc) {
return json -> {
HttpServerResponse response = rc.response()
.putHeader(HttpHeaders.CONTENT_TYPE, "application/json;charset=UTF-8");
if (json.failed()) {
if (json.cause().getMessage().toLowerCase().contains("not found")) {
response.setStatusCode(404);
} else {
response.setStatusCode(400);
}
response.end("{\"message\": \"" + json.cause().getMessage() + "\"}");
} else {
buildResponse(json.result(), response);
}
};
}
private void buildResponse(CheckResult json, HttpServerResponse response) {
int status = isUp(json) ? 200 : 503;
if (status == 503 && hasProcedureError(json)) {
status = 500;
}
List<CheckResult> checks = json.getChecks();
if (status == 200 && checks != null && checks.isEmpty()) {
response.setStatusCode(204).end();
return;
}
response
.setStatusCode(status)
.end(json.toJson().encode());
}
@Override
public synchronized HealthCheckHandler unregister(String name) {
healthChecks.unregister(name);
return this;
}
private boolean hasProcedureError(CheckResult json) {
JsonObject data = json.getData();
if (data != null && data.getBoolean("procedure-execution-failure", false)) {
return true;
}
List<CheckResult> checks = json.getChecks();
if (checks != null) {
for (CheckResult check : checks) {
if (hasProcedureError(check)) {
return true;
}
}
}
return false;
}
}