com.arpnetworking.clusteraggregator.http.Routes Maven / Gradle / Ivy
/*
* Copyright 2014 Groupon.com
*
* 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 com.arpnetworking.clusteraggregator.http;
import com.arpnetworking.clusteraggregator.Status;
import com.arpnetworking.clusteraggregator.models.StatusResponse;
import com.arpnetworking.clusteraggregator.models.VersionInfo;
import com.arpnetworking.commons.jackson.databind.ObjectMapperFactory;
import com.arpnetworking.configuration.jackson.pekko.PekkoModule;
import com.arpnetworking.metrics.incubator.PeriodicMetrics;
import com.arpnetworking.steno.LogBuilder;
import com.arpnetworking.steno.Logger;
import com.arpnetworking.steno.LoggerFactory;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.google.common.collect.ImmutableList;
import com.google.inject.Inject;
import com.google.inject.Singleton;
import com.google.inject.name.Named;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
import org.apache.pekko.actor.ActorSystem;
import org.apache.pekko.cluster.Member;
import org.apache.pekko.http.javadsl.model.ContentType;
import org.apache.pekko.http.javadsl.model.ContentTypes;
import org.apache.pekko.http.javadsl.model.HttpHeader;
import org.apache.pekko.http.javadsl.model.HttpMethods;
import org.apache.pekko.http.javadsl.model.HttpRequest;
import org.apache.pekko.http.javadsl.model.HttpResponse;
import org.apache.pekko.http.javadsl.model.StatusCodes;
import org.apache.pekko.http.javadsl.model.headers.CacheControl;
import org.apache.pekko.http.javadsl.model.headers.CacheDirectives;
import org.apache.pekko.japi.function.Function;
import org.apache.pekko.pattern.Patterns;
import org.apache.pekko.util.ByteString;
import java.io.IOException;
import java.time.Duration;
import java.util.Optional;
import java.util.UUID;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionStage;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;
import javax.annotation.Nullable;
/**
* Http server routes.
*
* @author Ville Koskela (ville dot koskela at inscopemetrics dot com)
*/
@Singleton
public final class Routes implements Function> {
/**
* Public constructor.
*
* @param actorSystem Instance of {@link ActorSystem}.
* @param periodicMetrics Instance of {@link PeriodicMetrics}.
* @param healthCheckPath The path for the health check.
* @param statusPath The path for the status.
* @param versionPath The path for the version.
*/
@Inject
public Routes(
final ActorSystem actorSystem,
final PeriodicMetrics periodicMetrics,
@Named("health-check-path")
final String healthCheckPath,
@Named("status-path")
final String statusPath,
@Named("version-path")
final String versionPath) {
_actorSystem = actorSystem;
_periodicMetrics = periodicMetrics;
_healthCheckPath = healthCheckPath;
_statusPath = statusPath;
_versionPath = versionPath;
_objectMapper = ObjectMapperFactory.createInstance();
_objectMapper.registerModule(new SimpleModule().addSerializer(Member.class, new MemberSerializer()));
_objectMapper.registerModule(new PekkoModule(actorSystem));
}
@Override
public CompletionStage apply(final HttpRequest request) {
final long requestStartTime = System.currentTimeMillis();
final UUID requestId = UUID.randomUUID();
if (LOGGER.isTraceEnabled()) {
LOGGER.trace()
.setEvent("http.in.start")
.addContext("requestId", requestId)
.addData("method", request.method().toString())
.addData("url", request.getUri().toString())
.addData(
"headers",
StreamSupport.stream(request.getHeaders().spliterator(), false)
.map(h -> h.name() + "=" + h.value())
.collect(Collectors.toList()))
.log();
}
return process(request).whenComplete(
(response, failure) -> {
final int responseStatus;
if (response != null) {
responseStatus = response.status().intValue();
} else {
// TODO(ville): Figure out how to intercept post-exception mapping.
responseStatus = 599;
}
_periodicMetrics.recordTimer(
createMetricName(request, responseStatus, REQUEST_METRIC),
System.currentTimeMillis() - requestStartTime,
Optional.of(TimeUnit.MILLISECONDS));
_periodicMetrics.recordGauge(
createMetricName(request, responseStatus, BODY_SIZE_METRIC),
request.entity().getContentLengthOption().orElse(0L));
final int responseStatusClass = responseStatus / 100;
for (final int i : STATUS_CLASSES) {
_periodicMetrics.recordCounter(
createMetricName(request, responseStatus, String.format("%s/%dxx", STATUS_METRIC, i)),
responseStatusClass == i ? 1 : 0);
}
final LogBuilder log;
if (failure != null || responseStatusClass == 5) {
log = LOGGER.info().setEvent("http.in.failure");
if (failure != null) {
log.setThrowable(failure);
}
if (!LOGGER.isTraceEnabled() && LOGGER.isInfoEnabled()) {
log.addData("method", request.method().toString())
.addData("url", request.getUri().toString())
.addData(
"headers",
StreamSupport.stream(request.getHeaders().spliterator(), false)
.map(h -> h.name() + "=" + h.value())
.collect(Collectors.toList()));
}
} else {
log = LOGGER.trace().setEvent("http.in.complete");
}
log.addContext("requestId", requestId)
.addData("status", responseStatus)
.log();
});
}
/**
* Start failing health checks in preparation for service shutdown.
*/
public void shutdownHealthcheck() {
_healthCheckShutdown.set(true);
}
private CompletionStage process(final HttpRequest request) {
if (HttpMethods.GET.equals(request.method())) {
if (_healthCheckPath.equals(request.getUri().path())) {
if (!_healthCheckShutdown.get()) {
return ask("/user/status", new Status.HealthRequest(), Boolean.FALSE)
.thenApply(
isHealthy -> HttpResponse.create()
.withStatus(isHealthy ? StatusCodes.OK : StatusCodes.INTERNAL_SERVER_ERROR)
.addHeader(PING_CACHE_CONTROL_HEADER)
.withEntity(
JSON_CONTENT_TYPE,
ByteString.fromString(
"{\"status\":\""
+ (isHealthy ? HEALTHY_STATE : UNHEALTHY_STATE)
+ "\"}")));
} else {
return CompletableFuture.completedFuture(HttpResponse.create()
.withStatus(StatusCodes.INTERNAL_SERVER_ERROR)
.addHeader(PING_CACHE_CONTROL_HEADER)
.withEntity(
JSON_CONTENT_TYPE,
ByteString.fromString(
"{\"status\":\""
+ UNHEALTHY_STATE
+ "\"}")));
}
} else if (_statusPath.equals(request.getUri().path())) {
return ask("/user/status", new Status.StatusRequest(), (StatusResponse) null)
.thenApply(
status -> {
try {
return HttpResponse.create()
.withEntity(
JSON_CONTENT_TYPE,
ByteString.fromString(_objectMapper.writeValueAsString(status)));
} catch (final IOException e) {
LOGGER.error()
.setMessage("Failed to serialize status")
.setThrowable(e)
.log();
return HttpResponse.create().withStatus(StatusCodes.INTERNAL_SERVER_ERROR);
}
});
} else if (_versionPath.equals(request.getUri().path())) {
return CompletableFuture.supplyAsync(() -> {
try {
return HttpResponse.create()
.withEntity(
JSON_CONTENT_TYPE,
ByteString.fromString(_objectMapper.writeValueAsString(VersionInfo.getInstance())));
} catch (final IOException e) {
LOGGER.error()
.setMessage("Failed to serialize version")
.setThrowable(e)
.log();
return HttpResponse.create().withStatus(StatusCodes.INTERNAL_SERVER_ERROR);
}
});
}
} else if (HttpMethods.POST.equals(request.method())) {
if (INCOMING_DATA_V1_PATH.equals(request.getUri().path())
|| INCOMING_DATA_PERSIST_V1_PATH.equals(request.getUri().path())
|| INCOMING_DATA_REAGGREGATE_V1_PATH.equals(request.getUri().path())) {
return ask("/user/http-ingest-v1", request, HttpResponse.create().withStatus(500));
}
}
return CompletableFuture.completedFuture(HttpResponse.create().withStatus(404));
}
@SuppressWarnings("unchecked")
private CompletionStage ask(final String actorPath, final Object request, @Nullable final T defaultValue) {
return
Patterns.ask(
_actorSystem.actorSelection(actorPath),
request,
Duration.ofSeconds(5))
.thenApply(o -> (T) o)
.exceptionally(throwable -> {
LOGGER.error()
.setMessage("error when routing ask")
.addData("actorPath", actorPath)
.addData("request", request)
.setThrowable(throwable)
.log();
return defaultValue;
});
}
private String createMetricName(final HttpRequest request, final int responseStatus, final String actionPart) {
final StringBuilder nameBuilder = new StringBuilder()
.append(REST_SERVICE_METRIC_ROOT)
.append(request.method().value());
if (responseStatus == 404) {
nameBuilder.append("/unknown_route");
} else {
if (!request.getUri().path().startsWith("/")) {
nameBuilder.append("/");
}
nameBuilder.append(request.getUri().path());
}
nameBuilder.append("/");
nameBuilder.append(actionPart);
return nameBuilder.toString();
}
@SuppressFBWarnings("SE_TRANSIENT_FIELD_NOT_RESTORED")
private final transient ActorSystem _actorSystem;
@SuppressFBWarnings("SE_TRANSIENT_FIELD_NOT_RESTORED")
private final transient PeriodicMetrics _periodicMetrics;
private final String _healthCheckPath;
private final String _statusPath;
private final String _versionPath;
private final ObjectMapper _objectMapper;
private AtomicBoolean _healthCheckShutdown = new AtomicBoolean(false);
private static final Logger LOGGER = LoggerFactory.getLogger(Routes.class);
private static final HttpHeader PING_CACHE_CONTROL_HEADER = CacheControl.create(
CacheDirectives.PRIVATE(),
CacheDirectives.NO_CACHE,
CacheDirectives.NO_STORE,
CacheDirectives.MUST_REVALIDATE);
private static final String UNHEALTHY_STATE = "UNHEALTHY";
private static final String HEALTHY_STATE = "HEALTHY";
private static final String REST_SERVICE_METRIC_ROOT = "rest_service/";
private static final String BODY_SIZE_METRIC = "body_size";
private static final String REQUEST_METRIC = "request";
private static final String STATUS_METRIC = "status";
private static final ImmutableList STATUS_CLASSES = ImmutableList.of(2, 3, 4, 5);
private static final ContentType JSON_CONTENT_TYPE = ContentTypes.APPLICATION_JSON;
private static final long serialVersionUID = -1573473630801540757L;
/**
* The legacy path for incoming data over HTTP. The reaggregation behavior
* on this path is controlled by the the {@code calculateClusterAggregations}
* value. If this is set to {@code False} this endpoint behaves with
* {@link com.arpnetworking.clusteraggregator.models.AggregationMode} set to
* {@code PERSIST}. If the configuration key is set to {@code True} this endpoint
* behaves with {@link com.arpnetworking.clusteraggregator.models.AggregationMode}
* set to {@code PERSIST_AND_REAGGREGATE}.
*/
public static final String INCOMING_DATA_V1_PATH = "/metrics/v1/data";
/**
* This HTTP endpoint is for only persisting data directly and through the
* host emitter.
*/
public static final String INCOMING_DATA_PERSIST_V1_PATH = "/metrics/v1/data/persist";
/**
* This HTTP endpoint is for only reaggregating data and then persisting
* through the cluster emitter.
*/
public static final String INCOMING_DATA_REAGGREGATE_V1_PATH = "/metrics/v1/data/reaggregate";
private static final class MemberSerializer extends JsonSerializer {
@Override
public void serialize(
final Member value,
final JsonGenerator jgen,
final SerializerProvider provider) throws IOException {
jgen.writeStartObject();
jgen.writeObjectField("address", value.address().toString());
jgen.writeObjectField("roles", value.getRoles().toArray());
jgen.writeEndObject();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy