All Downloads are FREE. Search and download functionalities are using the official Maven repository.

datahub.client.rest.RestEmitter Maven / Gradle / Ivy

There is a newer version: 0.15.0rc25
Show newest version
package datahub.client.rest;

import static com.linkedin.metadata.Constants.*;

import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.core.StreamReadConstraints;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.annotations.VisibleForTesting;
import com.linkedin.data.DataMap;
import com.linkedin.data.template.JacksonDataTemplateCodec;
import com.linkedin.mxe.MetadataChangeProposal;
import datahub.client.Callback;
import datahub.client.Emitter;
import datahub.client.MetadataResponseFuture;
import datahub.client.MetadataWriteResponse;
import datahub.event.EventFormatter;
import datahub.event.MetadataChangeProposalWrapper;
import datahub.event.UpsertAspectRequest;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.security.KeyManagementException;
import java.security.KeyStoreException;
import java.security.NoSuchAlgorithmException;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.Consumer;
import javax.annotation.concurrent.ThreadSafe;
import javax.net.ssl.SSLContext;
import lombok.extern.slf4j.Slf4j;
import org.apache.hc.client5.http.async.methods.SimpleHttpRequest;
import org.apache.hc.client5.http.async.methods.SimpleHttpResponse;
import org.apache.hc.client5.http.async.methods.SimpleRequestBuilder;
import org.apache.hc.client5.http.config.RequestConfig;
import org.apache.hc.client5.http.config.TlsConfig;
import org.apache.hc.client5.http.impl.async.CloseableHttpAsyncClient;
import org.apache.hc.client5.http.impl.async.HttpAsyncClientBuilder;
import org.apache.hc.client5.http.impl.nio.PoolingAsyncClientConnectionManagerBuilder;
import org.apache.hc.client5.http.ssl.ClientTlsStrategyBuilder;
import org.apache.hc.client5.http.ssl.NoopHostnameVerifier;
import org.apache.hc.client5.http.ssl.TrustAllStrategy;
import org.apache.hc.core5.concurrent.FutureCallback;
import org.apache.hc.core5.http.ContentType;
import org.apache.hc.core5.http.HttpStatus;
import org.apache.hc.core5.http.nio.ssl.TlsStrategy;
import org.apache.hc.core5.http2.HttpVersionPolicy;
import org.apache.hc.core5.ssl.SSLContexts;
import org.apache.hc.core5.util.TimeValue;

@ThreadSafe
@Slf4j
/**
 * The REST emitter is a thin wrapper on top of the Apache HttpClient
 * (https://hc.apache.org/httpcomponents-client-4.5.x/index.html) library. It supports non-blocking
 * emission of metadata and handles the details of JSON serialization of metadata aspects over the
 * wire.
 *
 * 

Constructing a REST Emitter follows a lambda-based fluent builder pattern using the `create` * method. e.g. RestEmitter emitter = RestEmitter.create(b :: b .server("http://localhost:8080") * .extraHeaders(Collections.singletonMap("Custom-Header", "custom-val") ); You can also customize * the underlying http client by calling the `customizeHttpAsyncClient` method on the builder. e.g. * RestEmitter emitter = RestEmitter.create(b :: b .server("http://localhost:8080") * .extraHeaders(Collections.singletonMap("Custom-Header", "custom-val") .customizeHttpAsyncClient(c * :: c.setConnectionTimeToLive(30, TimeUnit.SECONDS)) ); */ public class RestEmitter implements Emitter { private final RestEmitterConfig config; private final String ingestProposalUrl; private final String ingestOpenApiUrl; private final String configUrl; private final ObjectMapper objectMapper; private final JacksonDataTemplateCodec dataTemplateCodec; private final CloseableHttpAsyncClient httpClient; private final EventFormatter eventFormatter; /** * The default constructor, prefer using the `create` factory method. * * @param config */ public RestEmitter(RestEmitterConfig config) { objectMapper = new ObjectMapper().setSerializationInclusion(JsonInclude.Include.NON_NULL); int maxSize = Integer.parseInt( System.getenv() .getOrDefault(INGESTION_MAX_SERIALIZED_STRING_LENGTH, MAX_JACKSON_STRING_SIZE)); objectMapper .getFactory() .setStreamReadConstraints(StreamReadConstraints.builder().maxStringLength(maxSize).build()); dataTemplateCodec = new JacksonDataTemplateCodec(objectMapper.getFactory()); this.config = config; HttpAsyncClientBuilder httpClientBuilder = this.config.getAsyncHttpClientBuilder(); httpClientBuilder.setRetryStrategy(new DatahubHttpRequestRetryStrategy()); // Override httpClient settings with RestEmitter configs if present if (config.getTimeoutSec() != null) { httpClientBuilder.setDefaultRequestConfig( RequestConfig.custom() .setConnectionRequestTimeout( config.getTimeoutSec() * 1000, java.util.concurrent.TimeUnit.MILLISECONDS) .setResponseTimeout( config.getTimeoutSec() * 1000, java.util.concurrent.TimeUnit.MILLISECONDS) .build()); } PoolingAsyncClientConnectionManagerBuilder poolingAsyncClientConnectionManagerBuilder = PoolingAsyncClientConnectionManagerBuilder.create(); // Forcing http 1.x as 2.0 is not supported yet TlsConfig tlsHttp1Config = TlsConfig.copy(TlsConfig.DEFAULT).setVersionPolicy(HttpVersionPolicy.FORCE_HTTP_1).build(); poolingAsyncClientConnectionManagerBuilder.setDefaultTlsConfig(tlsHttp1Config); if (config.isDisableSslVerification()) { try { SSLContext sslcontext = SSLContexts.custom().loadTrustMaterial(TrustAllStrategy.INSTANCE).build(); TlsStrategy tlsStrategy = ClientTlsStrategyBuilder.create() .setSslContext(sslcontext) .setHostnameVerifier(NoopHostnameVerifier.INSTANCE) .build(); poolingAsyncClientConnectionManagerBuilder.setTlsStrategy(tlsStrategy); } catch (KeyManagementException | NoSuchAlgorithmException | KeyStoreException e) { throw new RuntimeException("Error while creating insecure http client", e); } } httpClientBuilder.setConnectionManager(poolingAsyncClientConnectionManagerBuilder.build()); httpClientBuilder.setRetryStrategy( new DatahubHttpRequestRetryStrategy( config.getMaxRetries(), TimeValue.ofSeconds(config.getRetryIntervalSec()))); this.httpClient = httpClientBuilder.build(); this.httpClient.start(); this.ingestProposalUrl = this.config.getServer() + "/aspects?action=ingestProposal"; this.ingestOpenApiUrl = config.getServer() + "/openapi/entities/v1/"; this.configUrl = this.config.getServer() + "/config"; this.eventFormatter = this.config.getEventFormatter(); } private static MetadataWriteResponse mapResponse(SimpleHttpResponse response) { MetadataWriteResponse.MetadataWriteResponseBuilder builder = MetadataWriteResponse.builder().underlyingResponse(response); if ((response != null) && (response.getCode()) == HttpStatus.SC_OK || Objects.requireNonNull(response).getCode() == HttpStatus.SC_CREATED) { builder.success(true); } else { builder.success(false); } // Read response content try { ByteArrayOutputStream result = new ByteArrayOutputStream(); builder.responseContent(response.getBody().getBodyText()); } catch (Exception e) { // Catch all exceptions and still return a valid response object log.warn("Wasn't able to convert response into a string", e); } return builder.build(); } /** * Constructing a REST Emitter follows a lambda-based fluent builder pattern using the `create` * method. e.g. RestEmitter emitter = RestEmitter.create(b :: b .server("http://localhost:8080") * // coordinates of gms server .extraHeaders(Collections.singletonMap("Custom-Header", * "custom-val") ); You can also customize the underlying http client by calling the * `customizeHttpAsyncClient` method on the builder. e.g. RestEmitter emitter = * RestEmitter.create(b :: b .server("http://localhost:8080") * .extraHeaders(Collections.singletonMap("Custom-Header", "custom-val") * .customizeHttpAsyncClient(c :: c.setConnectionTimeToLive(30, TimeUnit.SECONDS)) ); * * @param builderSupplier * @return a constructed RestEmitter. Call #testConnection to make sure this emitter has a valid * connection to the server */ public static RestEmitter create( Consumer builderSupplier) { RestEmitter restEmitter = new RestEmitter(RestEmitterConfig.builder().with(builderSupplier).build()); return restEmitter; } /** * Creates a RestEmitter with default settings. * * @return a constructed RestEmitter. Call #test_connection to validate that this emitter can * communicate with the server. */ public static RestEmitter createWithDefaults() { // No-op creator -> creates RestEmitter using default settings return create(b -> {}); } @Override public Future emit(MetadataChangeProposalWrapper mcpw, Callback callback) throws IOException { return emit(this.eventFormatter.convert(mcpw), callback); } @Override public Future emit(MetadataChangeProposal mcp, Callback callback) throws IOException { DataMap map = new DataMap(); map.put("proposal", mcp.data()); String serializedMCP = dataTemplateCodec.mapToString(map); log.debug("Emit: URL: {}, Payload: {}\n", this.ingestProposalUrl, serializedMCP); return this.postGeneric(this.ingestProposalUrl, serializedMCP, mcp, callback); } private Future postGeneric( String urlStr, String payloadJson, Object originalRequest, Callback callback) throws IOException { SimpleRequestBuilder simpleRequestBuilder = SimpleRequestBuilder.post(urlStr); simpleRequestBuilder.setHeader("Content-Type", "application/json"); simpleRequestBuilder.setHeader("X-RestLi-Protocol-Version", "2.0.0"); simpleRequestBuilder.setHeader("Accept", "application/json"); this.config.getExtraHeaders().forEach(simpleRequestBuilder::setHeader); if (this.config.getToken() != null) { simpleRequestBuilder.setHeader("Authorization", "Bearer " + this.config.getToken()); } simpleRequestBuilder.setBody(payloadJson, ContentType.APPLICATION_JSON); AtomicReference responseAtomicReference = new AtomicReference<>(); CountDownLatch responseLatch = new CountDownLatch(1); FutureCallback httpCallback = new FutureCallback() { @Override public void completed(SimpleHttpResponse response) { MetadataWriteResponse writeResponse = null; try { writeResponse = mapResponse(response); responseAtomicReference.set(writeResponse); } catch (Exception e) { // do nothing } responseLatch.countDown(); if (callback != null) { try { callback.onCompletion(writeResponse); } catch (Exception e) { log.error("Error executing user callback on completion.", e); } } } @Override public void failed(Exception ex) { responseLatch.countDown(); if (callback != null) { try { callback.onFailure(ex); } catch (Exception e) { log.error("Error executing user callback on failure.", e); } } } @Override public void cancelled() { responseLatch.countDown(); if (callback != null) { try { callback.onFailure(new RuntimeException("Cancelled")); } catch (Exception e) { log.error("Error executing user callback on failure due to cancellation.", e); } } } }; Future requestFuture = httpClient.execute(simpleRequestBuilder.build(), httpCallback); return new MetadataResponseFuture(requestFuture, responseAtomicReference, responseLatch); } private Future getGeneric(String urlStr) throws IOException { SimpleHttpRequest simpleHttpRequest = SimpleRequestBuilder.get(urlStr) .addHeader("Content-Type", "application/json") .addHeader("X-RestLi-Protocol-Version", "2.0.0") .addHeader("Accept", "application/json") .build(); Future response = this.httpClient.execute(simpleHttpRequest, null); return new MetadataResponseFuture(response, RestEmitter::mapResponse); } @Override public boolean testConnection() throws IOException, ExecutionException, InterruptedException { return this.getGeneric(this.configUrl).get().isSuccess(); } @Override public void close() throws IOException { this.httpClient.close(); } @Override public Future emit(List request, Callback callback) throws IOException { log.debug("Emit: URL: {}, Payload: {}\n", this.ingestOpenApiUrl, request); return this.postOpenAPI(request, callback); } private Future postOpenAPI( List payload, Callback callback) throws IOException { SimpleRequestBuilder simpleRequestBuilder = SimpleRequestBuilder.post(ingestOpenApiUrl) .addHeader("Content-Type", "application/json") .addHeader("Accept", "application/json") .addHeader("X-RestLi-Protocol-Version", "2.0.0"); this.config.getExtraHeaders().forEach(simpleRequestBuilder::addHeader); if (this.config.getToken() != null) { simpleRequestBuilder.addHeader("Authorization", "Bearer " + this.config.getToken()); } simpleRequestBuilder.setBody( objectMapper.writeValueAsString(payload), ContentType.APPLICATION_JSON); AtomicReference responseAtomicReference = new AtomicReference<>(); CountDownLatch responseLatch = new CountDownLatch(1); FutureCallback httpCallback = new FutureCallback() { @Override public void completed(SimpleHttpResponse response) { MetadataWriteResponse writeResponse = null; try { writeResponse = mapResponse(response); responseAtomicReference.set(writeResponse); } catch (Exception e) { // do nothing } responseLatch.countDown(); if (callback != null) { try { callback.onCompletion(writeResponse); } catch (Exception e) { log.error("Error executing user callback on completion.", e); } } } @Override public void failed(Exception ex) { responseLatch.countDown(); if (callback != null) { try { callback.onFailure(ex); } catch (Exception e) { log.error("Error executing user callback on failure.", e); } } } @Override public void cancelled() { responseLatch.countDown(); if (callback != null) { try { callback.onFailure(new RuntimeException("Cancelled")); } catch (Exception e) { log.error("Error executing user callback on failure due to cancellation.", e); } } } }; Future requestFuture = httpClient.execute(simpleRequestBuilder.build(), httpCallback); return new MetadataResponseFuture(requestFuture, responseAtomicReference, responseLatch); } @VisibleForTesting CloseableHttpAsyncClient getHttpClient() { return this.httpClient; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy