io.gravitee.elasticsearch.client.http.HttpClient Maven / Gradle / Ivy
/**
* Copyright (C) 2015 The Gravitee team (http://gravitee.io)
*
* 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 io.gravitee.elasticsearch.client.http;
import com.fasterxml.jackson.databind.ObjectMapper;
import io.gravitee.common.http.HttpHeaders;
import io.gravitee.common.http.HttpStatusCode;
import io.gravitee.common.http.MediaType;
import io.gravitee.elasticsearch.client.Client;
import io.gravitee.elasticsearch.config.Endpoint;
import io.gravitee.elasticsearch.exception.ElasticsearchException;
import io.gravitee.elasticsearch.model.Health;
import io.gravitee.elasticsearch.model.CountResponse;
import io.gravitee.elasticsearch.model.Response;
import io.gravitee.elasticsearch.model.SearchResponse;
import io.gravitee.elasticsearch.model.bulk.BulkResponse;
import io.gravitee.elasticsearch.version.ElasticsearchInfo;
import io.reactivex.Completable;
import io.reactivex.Single;
import io.vertx.core.Handler;
import io.vertx.ext.web.client.WebClientOptions;
import io.vertx.ext.web.client.impl.HttpContext;
import io.vertx.ext.web.client.impl.WebClientInternal;
import io.vertx.reactivex.core.Vertx;
import io.vertx.reactivex.core.buffer.Buffer;
import io.vertx.reactivex.ext.web.client.WebClient;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import javax.annotation.PostConstruct;
import java.net.URI;
import java.nio.charset.StandardCharsets;
import java.util.Base64;
import java.util.List;
import static java.lang.String.format;
/**
* @author David BRASSELY (david.brassely at graviteesource.com)
* @author Nicolas GERAUD (nicolas.geraud at graviteesource.com)
* @author GraviteeSource Team
*/
public class HttpClient implements Client {
/**
* Logger.
*/
private final Logger logger = LoggerFactory.getLogger(HttpClient.class);
private static final String HTTPS_SCHEME = "https";
private static final String CONTENT_TYPE = MediaType.APPLICATION_JSON + ";charset=UTF-8";
private static String URL_ROOT;
private static String URL_STATE_CLUSTER;
private static String URL_BULK;
private static String URL_TEMPLATE;
private static String URL_INGEST;
private static String URL_SEARCH;
private static String URL_COUNT;
@Autowired
private Vertx vertx;
/**
* Configuration of Elasticsearch (cluster name, addresses, ...)
*/
private HttpClientConfiguration configuration;
/**
* HTTP client.
*/
private WebClient httpClient;
/**
* Authorization header if Elasticsearch is protected.
*/
private String authorizationHeader;
private final ObjectMapper mapper = new ObjectMapper();
public HttpClient() {
this(new HttpClientConfiguration());
}
public HttpClient(final HttpClientConfiguration configuration) {
this.configuration = configuration;
}
@PostConstruct
public void initialize() {
if (! configuration.getEndpoints().isEmpty()) {
final Endpoint endpoint = configuration.getEndpoints().get(0);
final URI elasticEdpt = URI.create(endpoint.getUrl());
initializePaths(elasticEdpt);
WebClientOptions options = new WebClientOptions()
.setDefaultHost(elasticEdpt.getHost())
.setDefaultPort(elasticEdpt.getPort() != -1 ? elasticEdpt.getPort() :
(HTTPS_SCHEME.equals(elasticEdpt.getScheme()) ? 443 : 80));
if (HTTPS_SCHEME.equals(elasticEdpt.getScheme())) {
options
.setSsl(true)
.setTrustAll(true);
if (this.configuration.getSslConfig() != null) {
options.setKeyCertOptions(this.configuration.getSslConfig().getVertxWebClientSslKeystoreOptions());
}
}
this.httpClient = WebClient.create(vertx, options);
// Read configuration to authenticate calls to Elasticsearch (basic authentication only)
if (this.configuration.getUsername() != null) {
this.authorizationHeader = this.initEncodedAuthorization(this.configuration.getUsername(),
this.configuration.getPassword());
}
((WebClientInternal) this.httpClient.getDelegate()).addInterceptor(context -> {
context.request()
.timeout(configuration.getRequestTimeout())
.putHeader(HttpHeaders.ACCEPT, CONTENT_TYPE)
.putHeader(HttpHeaders.ACCEPT_CHARSET, StandardCharsets.UTF_8.name());
// Basic authentication
if (authorizationHeader != null) {
context.request().putHeader(HttpHeaders.AUTHORIZATION, authorizationHeader);
}
context.next();
});
}
}
private void initializePaths(URI uri) {
String urlPrefix = uri.getPath().replaceAll("/$", "");
URL_ROOT = urlPrefix + "/";
URL_STATE_CLUSTER = urlPrefix + "/_cluster/health";
URL_BULK = urlPrefix + "/_bulk";
URL_TEMPLATE = urlPrefix + "/_template";
URL_INGEST = urlPrefix + "/_ingest/pipeline";
URL_SEARCH = urlPrefix + "/_search?ignore_unavailable=true";
URL_COUNT = urlPrefix + "/_count?ignore_unavailable=true";
}
@Override
public Single getInfo() throws ElasticsearchException {
return httpClient
.get(URL_ROOT)
.rxSend()
.doOnError(throwable -> logger.error("Unable to get a connection to Elasticsearch", throwable))
.map(response -> mapper.readValue(response.bodyAsString(), ElasticsearchInfo.class));
}
/**
* Get the cluster health
*
* @return the cluster health
* @throws ElasticsearchException error occurs during ES call
*/
@Override
public Single getClusterHealth() {
return httpClient
.get(URL_STATE_CLUSTER)
.rxSend()
.map(response -> mapper.readValue(response.bodyAsString(), Health.class));
}
@Override
public Single bulk(final List data) {
// Compact buffer
Buffer payload = Buffer.buffer();
data.forEach(buffer -> payload.appendBuffer(Buffer.newInstance(buffer)));
return httpClient
.post(URL_BULK)
.putHeader(HttpHeaders.CONTENT_TYPE, "application/x-ndjson")
.rxSendBuffer(payload)
.map(response -> {
if (response.statusCode() != HttpStatusCode.OK_200) {
logger.error("Unable to bulk index data: status[{}] response[{}]",
response.statusCode(), response.body());
throw new ElasticsearchException("Unable to bulk index data");
}
BulkResponse bulkResponse = mapper.readValue(response.bodyAsString(), BulkResponse.class);
if (bulkResponse.getErrors()) {
bulkResponse.getItems().stream()
.filter(bulkItemResponse -> bulkItemResponse.getIndex().getError() != null)
.forEach(bulkItemResponse ->
logger.error("An error occurs while indexing data into ES: indice[{}] error[{}]",
bulkItemResponse.getIndex().getIndexName(),
bulkItemResponse.getIndex().getError().getReason()));
}
return bulkResponse;
});
}
@Override
public Completable putTemplate(String templateName, String template) {
return httpClient
.put(URL_TEMPLATE + '/' + templateName)
.putHeader(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON)
.rxSendBuffer(Buffer.buffer(template))
.flatMapCompletable(response -> {
if (response.statusCode() != HttpStatusCode.OK_200) {
logger.error("Unable to put template mapping: status[{}] template[{}] response[{}]",
response.statusCode(), template, response.body());
return Completable.error(new ElasticsearchException("Unable to put template mapping"));
}
return Completable.complete();
});
}
/**
* Perform an HTTP count query
* @param indexes indexes names. If null count on all indexes
* @param type document type separated by comma. If null count on all types
* @param query json body query
* @return elasticsearch response
*/
public Single count(final String indexes, final String type, final String query) {
// index can be null _count on all index
final StringBuilder url = new StringBuilder()
.append('/')
.append(indexes);
if (type != null) {
url.append('/').append(type);
}
url.append(URL_COUNT);
return httpClient
.post(url.toString())
.putHeader(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON)
.rxSendBuffer(Buffer.buffer(query))
.map(response -> {
if (response.statusCode() != HttpStatusCode.OK_200) {
logger.error("Unable to count: url[{}] status[{}] query[{}] response[{}]",
url.toString(), response.statusCode(), query, response.body());
throw new ElasticsearchException("Unable to count");
}
return mapper.readValue(response.bodyAsString(), CountResponse.class);
});
}
/**
* Perform an HTTP search query
* @param indexes indexes names. If null search on all indexes
* @param type document type separated by comma. If null search on all types
* @param query json body query
* @return elasticsearch response
*/
public Single search(final String indexes, final String type, final String query) {
// index can be null _search on all index
final StringBuilder url = new StringBuilder()
.append('/')
.append(indexes);
if (type != null) {
url.append('/').append(type);
}
url.append(URL_SEARCH);
return httpClient
.post(url.toString())
.putHeader(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON)
.rxSendBuffer(Buffer.buffer(query))
.map(response -> {
if (response.statusCode() != HttpStatusCode.OK_200) {
logger.error("Unable to search: url[{}] status[{}] query[{}] response[{}]",
url.toString(), response.statusCode(), query, response.body());
throw new ElasticsearchException("Unable to search");
}
return mapper.readValue(response.bodyAsString(), SearchResponse.class);
});
}
/**
* Perform an HTTP count query
* @param url URL to call
* @param query json body query
* @return elasticsearch response
*/
public Single count(final String url, final String query) {
return httpClient
.post(url)
.putHeader(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON)
.rxSendBuffer(Buffer.buffer(query))
.map(response -> {
if (response.statusCode() != HttpStatusCode.OK_200) {
logger.error("Unable to count: url[{}] status[{}] query[{}] response[{}]",
url, response.statusCode(), query, response.body());
throw new ElasticsearchException("Unable to count");
}
return mapper.readValue(response.bodyAsString(), CountResponse.class);
});
}
@Override
public Completable putPipeline(String pipelineName, String pipeline) {
return httpClient
.put(URL_INGEST + '/' + pipelineName)
.putHeader(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON)
.rxSendBuffer(Buffer.buffer(pipeline))
.flatMapCompletable(response -> {
switch (response.statusCode()) {
case HttpStatusCode.OK_200:
return Completable.complete();
case HttpStatusCode.BAD_REQUEST_400:
logger.warn("Unable to create ES pipeline: {}", pipelineName);
break;
default:
logger.error("Unable to put pipeline: status[{}] pipeline[{}] response[{}]",
response.statusCode(), pipeline, response.body());
break;
}
return Completable.error(new ElasticsearchException(
format("Unable to create ES pipeline '%s': status[%s] response[%s]",
pipelineName, response.statusCode(), response.body())));
});
}
/**
* Create the Basic HTTP auth
*
* @param username
* username
* @param password
* password
* @return Basic auth string
*/
private String initEncodedAuthorization(final String username, final String password) {
final String auth = username + ":" + password;
final String encodedAuth = Base64.getEncoder().encodeToString(auth.getBytes(StandardCharsets.UTF_8));
return "Basic " + encodedAuth;
}
public void setConfiguration(HttpClientConfiguration configuration) {
this.configuration = configuration;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy