ai.vespa.hosted.client.HttpClient Maven / Gradle / Ivy
// Copyright Vespa.ai. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
package ai.vespa.hosted.client;
import ai.vespa.http.HttpURL;
import ai.vespa.http.HttpURL.Path;
import ai.vespa.http.HttpURL.Query;
import ai.vespa.json.Json;
import com.yahoo.time.TimeBudget;
import org.apache.hc.client5.http.config.RequestConfig;
import org.apache.hc.core5.http.ClassicHttpRequest;
import org.apache.hc.core5.http.ClassicHttpResponse;
import org.apache.hc.core5.http.HttpEntity;
import org.apache.hc.core5.http.HttpStatus;
import org.apache.hc.core5.http.Method;
import org.apache.hc.core5.http.io.entity.EntityUtils;
import org.apache.hc.core5.util.Timeout;
import java.io.Closeable;
import java.io.IOException;
import java.io.InputStream;
import java.io.UncheckedIOException;
import java.net.URI;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.function.Function;
import java.util.function.Supplier;
import java.util.stream.IntStream;
import static java.nio.charset.StandardCharsets.UTF_8;
import static java.util.Objects.requireNonNull;
/**
* @author jonmv
*/
@SuppressWarnings("deprecation")
public interface HttpClient extends Closeable {
RequestConfig defaultRequestConfig = RequestConfig.custom()
.setConnectionRequestTimeout(Timeout.ofSeconds(5))
.setConnectTimeout(Timeout.ofSeconds(5))
.setRedirectsEnabled(false)
.build();
/** Does nothing, letting the client wrap with a {@link RetryException} and re-throw. */
ExceptionHandler retryAll = (exception, request) -> { };
/** Throws a a {@link RetryException} if {@code statusCode == 503}, or a {@link ResponseException} unless {@code 200 <= statusCode < 300}. */
ResponseVerifier throwOnError = new DefaultResponseVerifier() { };
/** Reads the response body, throwing an {@link UncheckedIOException} if this fails, or {@code null} if there is none. */
static byte[] getBytes(ClassicHttpResponse response) {
try {
return response.getEntity() == null ? null : EntityUtils.toByteArray(response.getEntity());
}
catch (IOException e) {
throw new UncheckedIOException(e);
}
}
/** Creates a builder for sending the given method, using the specified host strategy. */
RequestBuilder send(HostStrategy hosts, Method method);
/** Builder for a request against a given set of hosts, using this config server client. */
interface RequestBuilder {
/** Appends to the request path, with no trailing slash. */
default RequestBuilder at(String... pathSegments) { return at(List.of(pathSegments)); }
/** Appends to the request path, with no trailing slash. */
default RequestBuilder at(List pathSegments) { return at(Path.empty().append(pathSegments).withoutTrailingSlash()); }
/** Appends to the request path. */
RequestBuilder at(HttpURL.Path path);
/** Sets the request body as UTF-8 application/json. */
RequestBuilder body(byte[] json);
/** Sets the request body. */
RequestBuilder body(Supplier entity);
/** Sets query parameters without a value, like {@code ?debug&recursive}. */
default RequestBuilder emptyParameters(String... keys) {
return emptyParameters(List.of(keys));
}
/** Sets query parameters without a value, like {@code ?debug&recursive}. */
RequestBuilder emptyParameters(List keys);
/** Sets the parameter key/values for the request. Number of arguments must be even. Null values are omitted. */
default RequestBuilder parameters(String... pairs) {
return parameters(Arrays.asList(pairs));
}
/** Sets the parameter key/values for the request. Number of arguments must be even. Pairs with {@code null} values are omitted. */
RequestBuilder parameters(List pairs);
/** Appends all parameters from the given query. */
RequestBuilder parameters(Query query);
/** Sets all parameters from the given query dynamically, when creating retried requests. */
RequestBuilder parameters(Supplier query);
/** Overrides the default socket read timeout of the request. {@code Duration.ZERO} gives infinite timeout. */
RequestBuilder timeout(Duration timeout);
/**
* Pseudo-deadline for the request, including retries.
* Pseudo- because it only ensures request timeouts are low enough to honour the deadline, but nothing else.
*/
RequestBuilder deadline(TimeBudget deadline);
/** Adds the given header value to the list of headers with the given name. */
RequestBuilder addHeader(String name, String value);
/** Sets the list of headers with the given name to the given value. */
RequestBuilder setHeader(String name, String value);
/** Overrides the default request config of the request. */
RequestBuilder config(RequestConfig config);
/**
* Sets the catch clause for {@link IOException}s during execution of this.
* The default is to wrap the IOException in a {@link RetryException} and rethrow this;
* this makes the client retry the request, as long as there are remaining entries in the {@link HostStrategy}.
* If the catcher returns normally, the exception is wrapped and retried, as per the default.
*/
RequestBuilder catching(ExceptionHandler catcher);
/**
* Sets the (error) response handler for this request. The default is {@link #throwOnError}.
* When the handler returns normally, the response is treated as a success, and passed on to a response mapper.
*/
RequestBuilder throwing(ResponseVerifier handler);
/** Reads the response as a {@link String}, or throws if unsuccessful. */
String read();
/** Reads and maps the response, or throws if unsuccessful. */
T read(Function mapper);
/** Reads the response as a {@link Json}, or throws if unsuccessful. */
Json readAsJson();
/** Discards the response, but throws if unsuccessful. */
void discard();
/** Returns the raw response input stream, or throws if unsuccessful. The caller must close the returned stream. */
HttpInputStream stream();
/** Uses the response and request, if successful, to generate a mapped response. */
T handle(ResponseHandler handler);
}
class HttpInputStream extends ForwardingInputStream {
private final ClassicHttpResponse response;
protected HttpInputStream(ClassicHttpResponse response) throws IOException {
super(response.getEntity() != null ? response.getEntity().getContent()
: InputStream.nullInputStream());
this.response = response;
}
public int statusCode() { return response.getCode(); }
public String contentType() { return response.getEntity().getContentType(); }
@Override
public void close() throws IOException {
super.close();
response.close();
}
}
/** Reads a successful response and request to compute a result. */
@FunctionalInterface
interface ResponseHandler {
/** Called with successful responses, as per {@link ResponseVerifier}. The caller must close the response. */
T handle(ClassicHttpResponse response, ClassicHttpRequest request) throws IOException;
}
/** Verifies a response, throwing on error responses, possibly indicating retries. */
@FunctionalInterface
interface ResponseVerifier {
/** Whether this status code means the response is an error response. */
default boolean isError(int statusCode) {
return statusCode < HttpStatus.SC_OK || HttpStatus.SC_REDIRECTION <= statusCode;
}
/** Whether this status code means we should retry. Has no effect if this is not also an error. */
default boolean shouldRetry(int statusCode) {
return statusCode == HttpStatus.SC_SERVICE_UNAVAILABLE;
}
/** Verifies the given response, consuming it and throwing if it is an error; or leaving it otherwise. */
default void verify(ClassicHttpResponse response, ClassicHttpRequest request) throws IOException {
if (isError(response.getCode())) {
try (response) {
byte[] body = response.getEntity() == null ? new byte[0] : EntityUtils.toByteArray(response.getEntity());
RuntimeException exception = toException(response.getCode(), body, request);
throw shouldRetry(response.getCode()) ? new RetryException(exception) : exception;
}
}
}
/** Throws the appropriate exception, for the given status code and body. */
RuntimeException toException(int statusCode, byte[] body, ClassicHttpRequest request);
}
interface DefaultResponseVerifier extends ResponseVerifier {
@Override
default RuntimeException toException(int statusCode, byte[] body, ClassicHttpRequest request) {
return new ResponseException(statusCode,
request + " failed with status " + statusCode + " and body '" + new String(body, UTF_8) + "'");
}
}
@FunctionalInterface
interface ExceptionHandler {
/**
* Called with any IO exception that might occur when attempting to send the request.
* To retry, wrap the exception with a {@link RetryException} and re-throw, or exit normally.
* Any other thrown exception will propagate out to the caller.
*/
void handle(IOException exception, ClassicHttpRequest request);
}
/** What host(s) to try for a request, in what order. A host may be specified multiple times, for retries. */
@FunctionalInterface
interface HostStrategy extends Iterable {
/** Attempts the given host once. */
static HostStrategy of(URI host) {
return repeating(host, 1);
}
/** Attempts each request once against each listed host. */
static HostStrategy ordered(List hosts) {
return List.copyOf(hosts)::iterator;
}
/** Attempts each request once against each listed host, in random order. */
static HostStrategy shuffling(List hosts) {
return () -> {
List copy = new ArrayList<>(hosts);
Collections.shuffle(copy);
return copy.iterator();
};
}
/** Attempts each request against the host the specified number of times. */
static HostStrategy repeating(URI host, int count) {
return ordered(IntStream.range(0, count).mapToObj(__ -> host).toList());
}
}
/** Exception wrapper that signals retries should be attempted. */
final class RetryException extends RuntimeException {
public RetryException(IOException cause) {
super(requireNonNull(cause));
}
public RetryException(RuntimeException cause) {
super(requireNonNull(cause));
}
static RetryException wrap(IOException exception, ClassicHttpRequest request) {
return new RetryException(new UncheckedIOException(request + " failed (" + exception.getClass().getSimpleName() + ")", exception));
}
}
/** An exception due to server error, a bad request, or similar, which resulted in a non-OK HTTP response. */
class ResponseException extends RuntimeException {
private final int statusCode;
public ResponseException(int statusCode, String message) {
super(message);
this.statusCode = statusCode;
}
public int statusCode() { return statusCode; }
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy