com.palantir.dialogue.hc5.ApacheHttpClientBlockingChannel Maven / Gradle / Ivy
/*
* (c) Copyright 2020 Palantir Technologies Inc. All rights reserved.
*
* 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.palantir.dialogue.hc5;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ListMultimap;
import com.google.common.collect.MultimapBuilder;
import com.google.common.collect.Multimaps;
import com.google.common.io.ByteStreams;
import com.google.common.net.HttpHeaders;
import com.palantir.dialogue.Endpoint;
import com.palantir.dialogue.HttpMethod;
import com.palantir.dialogue.Request;
import com.palantir.dialogue.RequestBody;
import com.palantir.dialogue.Response;
import com.palantir.dialogue.ResponseAttachments;
import com.palantir.dialogue.blocking.BlockingChannel;
import com.palantir.dialogue.core.BaseUrl;
import com.palantir.logsafe.Arg;
import com.palantir.logsafe.Preconditions;
import com.palantir.logsafe.SafeArg;
import com.palantir.logsafe.SafeLoggable;
import com.palantir.logsafe.exceptions.SafeExceptions;
import com.palantir.logsafe.exceptions.SafeRuntimeException;
import com.palantir.logsafe.logger.SafeLogger;
import com.palantir.logsafe.logger.SafeLoggerFactory;
import com.palantir.tracing.api.TraceHttpHeaders;
import java.io.ByteArrayInputStream;
import java.io.FilterInputStream;
import java.io.FilterOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URL;
import java.util.Arrays;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.OptionalInt;
import java.util.OptionalLong;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import javax.annotation.Nullable;
import org.apache.hc.client5.http.ConnectTimeoutException;
import org.apache.hc.client5.http.classic.ExecRuntime;
import org.apache.hc.client5.http.impl.classic.CloseableHttpResponse;
import org.apache.hc.client5.http.protocol.HttpClientContext;
import org.apache.hc.core5.function.Supplier;
import org.apache.hc.core5.http.Header;
import org.apache.hc.core5.http.HttpEntity;
import org.apache.hc.core5.http.NoHttpResponseException;
import org.apache.hc.core5.http.io.support.ClassicRequestBuilder;
import org.apache.hc.core5.http.message.BasicHeader;
final class ApacheHttpClientBlockingChannel implements BlockingChannel {
private static final SafeLogger log = SafeLoggerFactory.get(ApacheHttpClientBlockingChannel.class);
/**
* Threshold beyond which connections are closed rather than attempting to exhaust the response content for reuse.
* Should be small enough that it's quick to load and check, but large enough that we don't cause unnecessary
* connection/handshake churn.
*/
private static final int REMAINING_CONTENT_CONNECTION_DISCARD_THRESHOLD = 64 * 1024;
private final ApacheHttpClientChannels.CloseableClient client;
private final BaseUrl baseUrl;
private final ResponseLeakDetector responseLeakDetector;
private final OptionalInt uriIndexForInstrumentation;
ApacheHttpClientBlockingChannel(
ApacheHttpClientChannels.CloseableClient client,
URL baseUrl,
ResponseLeakDetector responseLeakDetector,
OptionalInt uriIndexForInstrumentation) {
this.client = client;
this.baseUrl = BaseUrl.of(baseUrl);
this.responseLeakDetector = responseLeakDetector;
this.uriIndexForInstrumentation = uriIndexForInstrumentation;
}
@Override
public Response execute(Endpoint endpoint, Request request) throws IOException {
// Create base request given the URL
URL target = baseUrl.render(endpoint, request);
ClassicRequestBuilder builder =
ClassicRequestBuilder.create(endpoint.httpMethod().name()).setUri(target.toString());
// Fill headers
request.headerParams().forEach(builder::addHeader);
if (request.body().isPresent()) {
Preconditions.checkArgument(
endpoint.httpMethod() != HttpMethod.GET, "GET endpoints must not have a request body");
Preconditions.checkArgument(
endpoint.httpMethod() != HttpMethod.HEAD, "HEAD endpoints must not have a request body");
Preconditions.checkArgument(
endpoint.httpMethod() != HttpMethod.OPTIONS, "OPTIONS endpoints must not have a request body");
RequestBody body = request.body().get();
setBody(builder, body);
} else if (requiresEmptyBody(endpoint)) {
builder.setEntity(EmptyHttpEntity.INSTANCE);
}
long startTime = System.nanoTime();
try {
HttpClientContext context = HttpClientContext.create();
CloseableHttpResponse httpClientResponse = client.apacheClient().execute(builder.build(), context);
// Defensively ensure that resources are closed if failures occur within this block,
// for example HttpClientResponse allocation may throw an OutOfMemoryError.
boolean close = true;
try {
Response dialogueResponse = new HttpClientResponse(client, httpClientResponse, context);
Response leakDetectingResponse = responseLeakDetector.wrap(dialogueResponse, endpoint);
close = false;
return leakDetectingResponse;
} finally {
if (close) {
httpClientResponse.close();
}
}
} catch (ConnectTimeoutException e) {
// ConnectTimeoutException must be wrapped so it may be retried. SocketTimeoutExceptions are
// not retried by default, so ours implements SafeLoggable and retains the simple-name for
// cleaner metrics.
throw new SafeConnectTimeoutException(e, failureDiagnosticArgs(endpoint, request, startTime));
} catch (NoHttpResponseException e) {
// NoHttpResponseException may be thrown immediately when a request is sent if a pooled persistent
// connection has been closed by the target server, or an intermediate proxy. In this case it's
// important that we retry the request with a fresh connection.
// The other possibility is that a remote server or proxy may time out an active request due
// to inactivity and close the connection without a response, in this case the request mustn't
// be retried.
// We attempt to differentiate these two cases based on request duration, we expect most of
// the prior case to occur within a couple milliseconds, however we must use a larger value
// to account for large garbage collections.
long durationNanos = System.nanoTime() - startTime;
Arg>[] diagnosticArgs = failureDiagnosticArgs(endpoint, request, startTime);
if (durationNanos < TimeUnit.SECONDS.toNanos(5)) {
e.addSuppressed(new Diagnostic(diagnosticArgs));
throw e;
}
throw new SafeSocketTimeoutException("Received a NoHttpResponseException", e, diagnosticArgs);
} catch (Throwable t) {
// We can't wrap all potential exception types, that would cause the failure to lose some amount of type
// information. Instead, we add a suppressed throwable with no stack trace which acts as a courier
// for our diagnostic information, ensuring it can be recorded in the logs.
t.addSuppressed(new Diagnostic(failureDiagnosticArgs(endpoint, request, startTime)));
throw t;
}
}
private Arg>[] failureDiagnosticArgs(Endpoint endpoint, Request request, long startTimeNanos) {
long durationMillis = TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - startTimeNanos);
return new Arg>[] {
SafeArg.of("durationMillis", durationMillis),
SafeArg.of("connectTimeout", client.clientConfiguration().connectTimeout()),
SafeArg.of("socketTimeout", client.clientConfiguration().readTimeout()),
SafeArg.of("clientName", client.name()),
SafeArg.of("serviceName", endpoint.serviceName()),
SafeArg.of("endpointName", endpoint.endpointName()),
SafeArg.of("requestTraceId", request.headerParams().get(TraceHttpHeaders.TRACE_ID)),
// Request span ID can be used to associate a failed request with a request log line on the server.
SafeArg.of("requestSpanId", request.headerParams().get(TraceHttpHeaders.SPAN_ID)),
SafeArg.of("hostIndex", uriIndexForInstrumentation)
};
}
private static final class Diagnostic extends RuntimeException implements SafeLoggable {
private static final String SAFE_MESSAGE = "Client Failure Diagnostic Information";
private final List> args;
Diagnostic(Arg>[] args) {
super(SafeExceptions.renderMessage(SAFE_MESSAGE, args));
this.args = Collections.unmodifiableList(Arrays.asList(args));
}
@Override
public String getLogMessage() {
return SAFE_MESSAGE;
}
@Override
public List> getArgs() {
return args;
}
@Override
@SuppressWarnings("UnsynchronizedOverridesSynchronized") // nop
public Throwable fillInStackTrace() {
// no-op: stack trace generation is expensive, this type exists
// to simply associate diagnostic information with a failure.
return this;
}
}
// https://tools.ietf.org/html/rfc7230#section-3.3.2 recommends setting a content-length
// on empty post requests. Some components may respond 411 if the content-length is not present.
private static boolean requiresEmptyBody(Endpoint endpoint) {
HttpMethod method = endpoint.httpMethod();
return method == HttpMethod.POST || method == HttpMethod.PUT;
}
private static void setBody(ClassicRequestBuilder builder, RequestBody body) {
builder.setEntity(new RequestBodyEntity(body, contentLength(body, builder)));
}
private static OptionalLong contentLength(RequestBody requestBody, ClassicRequestBuilder builder) {
Header contentLengthHeader = builder.getFirstHeader(HttpHeaders.CONTENT_LENGTH);
OptionalLong headerContentLength = OptionalLong.empty();
if (contentLengthHeader != null) {
builder.removeHeaders(HttpHeaders.CONTENT_LENGTH);
String contentLengthValue = contentLengthHeader.getValue();
try {
headerContentLength = OptionalLong.of(Long.parseLong(contentLengthValue));
} catch (NumberFormatException nfe) {
log.warn(
"Failed to parse content-length value '{}'",
SafeArg.of(HttpHeaders.CONTENT_LENGTH, contentLengthValue),
nfe);
}
}
if (headerContentLength.isPresent() && requestBody.contentLength().isPresent()) {
long headerContentLengthValue = headerContentLength.getAsLong();
long requestBodyContentLength = requestBody.contentLength().getAsLong();
if (headerContentLengthValue != requestBodyContentLength) {
log.warn(
"Content lengths do not match",
SafeArg.of(HttpHeaders.CONTENT_LENGTH, headerContentLengthValue),
SafeArg.of("requestBodyContentLength", requestBodyContentLength));
}
}
if (headerContentLength.isPresent()) {
return headerContentLength;
}
return requestBody.contentLength();
}
private static final class HttpClientResponse implements Response {
private final CloseableHttpResponse response;
private final HttpClientContext context;
private final ResponseAttachments attachments = ResponseAttachments.create();
// Client reference is used to prevent premature termination
@Nullable
private ApacheHttpClientChannels.CloseableClient client;
@Nullable
private ListMultimap headers;
@Nullable
private InputStream responseBody;
HttpClientResponse(
ApacheHttpClientChannels.CloseableClient client,
CloseableHttpResponse response,
HttpClientContext context) {
this.client = client;
this.response = response;
this.context = context;
}
@Override
public InputStream body() {
InputStream snapshot = this.responseBody;
if (snapshot == null) {
snapshot = createResponseBody();
this.responseBody = snapshot;
}
return snapshot;
}
private InputStream createResponseBody() {
HttpEntity entity = response.getEntity();
if (entity != null) {
try {
return new ResponseInputStream(entity.getContent(), this);
} catch (IOException e) {
throw new SafeRuntimeException("Failed to get response stream", e);
}
}
return new ByteArrayInputStream(new byte[0]);
}
@Override
public int code() {
return response.getCode();
}
@Override
public ListMultimap headers() {
if (headers == null) {
ListMultimap tmpHeaders = MultimapBuilder.treeKeys(String.CASE_INSENSITIVE_ORDER)
.arrayListValues()
.build();
Iterator headerIterator = response.headerIterator();
while (headerIterator.hasNext()) {
Header header = headerIterator.next();
String value = header.getValue();
if (value != null) {
tmpHeaders.put(header.getName(), value);
}
}
headers = Multimaps.unmodifiableListMultimap(tmpHeaders);
}
return headers;
}
@Override
public Optional getFirstHeader(String header) {
return Optional.ofNullable(response.getFirstHeader(header)).map(Header::getValue);
}
@Override
public ResponseAttachments attachments() {
return attachments;
}
@Override
public void close() {
ApacheHttpClientChannels.CloseableClient clientSnapshot = client;
client = null;
// Avoid attempting to close a response that has already been closed.
if (clientSnapshot != null) {
try {
// Check if the response has been fully drained. If not, we close the connection rather than
// potentially reading massive data unnecessarily.
if (hasSubstantialRemainingData(response)) {
ExecRuntime runtime = HttpClientExecRuntimeAttributeInterceptor.get(context);
if (runtime != null) {
runtime.discardEndpoint();
// Constructing the new metrics component in the unexpected case is more efficient than
// creating the meter for hundreds of services which never hit this case.
DialogueClientMetrics.of(
clientSnapshot.clientConfiguration().taggedMetricRegistry())
.connectionClosedPartiallyConsumedResponse(clientSnapshot.name())
.mark();
// Do not call response.close which internally attempts to drain the response
// because the underlying resources have already been closed.
return;
}
}
response.close();
} catch (IOException | RuntimeException e) {
log.warn("Failed to close response", e);
}
}
}
boolean isOpen() {
return client != null;
}
@Override
public String toString() {
return "HttpClientResponse{response=" + response + ", client=" + client + '}';
}
}
/**
* Checks if there is remaining data in the stream, note that this is a
* destructive operation which should only occur in order to close the stream.
*/
private static boolean hasSubstantialRemainingData(CloseableHttpResponse response) {
try {
HttpEntity entity = response.getEntity();
if (entity == null || !entity.isStreaming()) {
return false;
}
InputStream stream = entity.getContent();
// Fast check: The stream has been fully exhausted in the expected case,
// no need to create buffers for drainage unless we know there's data to drain.
if (stream.read() == -1) {
return false;
}
return REMAINING_CONTENT_CONNECTION_DISCARD_THRESHOLD
== ByteStreams.exhaust(ByteStreams.limit(stream, REMAINING_CONTENT_CONNECTION_DISCARD_THRESHOLD));
} catch (Throwable ignored) {
return false;
}
}
private static final class RequestBodyEntity implements HttpEntity {
private final RequestBody requestBody;
private final Header contentType;
private final OptionalLong contentLength;
RequestBodyEntity(RequestBody requestBody, OptionalLong contentLength) {
this.requestBody = requestBody;
this.contentType = new BasicHeader(HttpHeaders.CONTENT_TYPE, requestBody.contentType());
this.contentLength = contentLength;
}
@Override
public boolean isRepeatable() {
// n.b. Proxy authentication can only be negotiated on repeatable requests.
// Subsequent requests needn't be repeatable as state is cached by the client.
return requestBody.repeatable();
}
@Override
public boolean isChunked() {
return !contentLength.isPresent();
}
@Override
public Set getTrailerNames() {
return Collections.emptySet();
}
@Override
public long getContentLength() {
return contentLength.orElse(-1);
}
@Override
public String getContentType() {
return contentType.getValue();
}
@Override
@Nullable
public String getContentEncoding() {
return null;
}
@Override
public InputStream getContent() throws UnsupportedOperationException {
throw new UnsupportedOperationException("getContent is not supported, writeTo should be used");
}
@Override
public void writeTo(OutputStream outStream) throws IOException {
requestBody.writeTo(new ModulatingOutputStream(outStream));
}
@Override
public boolean isStreaming() {
// Applies to responses.
return false;
}
@Override
@Nullable
public Supplier> getTrailers() {
return null;
}
@Override
public String toString() {
return "RequestBodyEntity{requestBody=" + requestBody + '}';
}
@Override
public void close() {}
}
/**
* {@link ModulatingOutputStream} limits the size of individual writes to the wrapped {@link OutputStream}
* in order to prevent degraded performance on large buffers as described in
* hadoop-crypto#586.
*/
static final class ModulatingOutputStream extends FilterOutputStream {
/**
* Block size of 16 KB is small enough to allow cipher implementations to become hot and optimize properly
* when given large inputs. Otherwise large array writes into a {@link javax.crypto.CipherOutputStream} fail to
* use intrinsified implementations. If 16 KB blocks aren't enough to produce hot methods, the I/O is small
* and infrequent enough that performance isn't relevant.
* For more information, see the details around {@code com.sun.crypto.provider.GHASH::processBlocks} in
*
* hadoop-crypto#586 (comment)
*/
private static final int BLOCK_SIZE = 16 * 1024;
ModulatingOutputStream(OutputStream delegate) {
super(delegate);
}
@Override
public void write(byte[] buffer, int off, int len) throws IOException {
Objects.checkFromIndexSize(off, len, buffer.length);
int currentOffset = off;
int remaining = len;
while (remaining > 0) {
int toWrite = Math.min(remaining, BLOCK_SIZE);
out.write(buffer, currentOffset, toWrite);
currentOffset += toWrite;
remaining -= toWrite;
}
}
@Override
public void write(int value) throws IOException {
out.write(value);
}
}
private static final class ResponseInputStream extends FilterInputStream {
private final HttpClientResponse response;
ResponseInputStream(InputStream stream, HttpClientResponse response) {
super(stream);
this.response = response;
}
@Override
public int read() throws IOException {
checkOpen();
return super.read();
}
@Override
public int read(byte[] buffer) throws IOException {
checkOpen();
return super.read(buffer);
}
@Override
public int read(byte[] buffer, int off, int len) throws IOException {
checkOpen();
return super.read(buffer, off, len);
}
@Override
public long skip(long num) throws IOException {
checkOpen();
return super.skip(num);
}
@Override
public void close() {
if (response.isOpen()) {
response.close();
// no need to close the delegate stream itself, closing the response is sufficient
// to release resources.
}
}
private void checkOpen() throws IOException {
if (!response.isOpen()) {
throw new DialogueStreamClosedException();
}
}
@Override
public String toString() {
return "ResponseInputStream{" + in + '}';
}
}
private static final class DialogueStreamClosedException extends IOException implements SafeLoggable {
private static final String MESSAGE = "Response has already been closed";
DialogueStreamClosedException() {
super(MESSAGE);
}
@Override
public String getLogMessage() {
return MESSAGE;
}
@Override
public List> getArgs() {
return ImmutableList.of();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy