com.aeontronix.enhancedmule.tools.util.HttpHelper Maven / Gradle / Ivy
/*
* Copyright (c) Aeontronix 2019
*/
package com.aeontronix.enhancedmule.tools.util;
import com.aeontronix.enhancedmule.tools.AnypointClient;
import com.aeontronix.enhancedmule.tools.authentication.AuthenticationProvider;
import com.aeontronix.enhancedmule.tools.authentication.AuthenticationProviderUsernamePasswordImpl;
import com.aeontronix.enhancedmule.tools.Environment;
import com.kloudtek.util.Base64;
import com.kloudtek.util.ThreadUtils;
import com.kloudtek.util.io.IOUtils;
import org.apache.http.HttpEntity;
import org.apache.http.HttpHeaders;
import org.apache.http.HttpHost;
import org.apache.http.client.methods.*;
import org.apache.http.entity.ByteArrayEntity;
import org.apache.http.entity.ContentType;
import org.apache.http.entity.mime.MultipartEntityBuilder;
import org.apache.http.impl.client.*;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.Closeable;
import java.io.IOException;
import java.io.OutputStream;
import java.nio.charset.StandardCharsets;
import java.util.HashMap;
import java.util.Map;
public class HttpHelper implements Closeable {
private static final Logger logger = LoggerFactory.getLogger(HttpHelper.class);
private static final String HEADER_AUTH = "Authorization";
private EMHttpClient httpClient;
private AuthenticationProvider authenticationProvider;
private String authToken;
private AnypointClient client;
private int maxRetries = 4;
private boolean loginRequest = false;
private int loginAttempts = 0;
private long retryDelay = 1000L;
public HttpHelper() {
}
public HttpHelper(AnypointClient client, AuthenticationProvider authenticationProvider) {
this.client = client;
this.httpClient = authenticationProvider.createHttpClient();
this.authenticationProvider = authenticationProvider;
}
@Override
public void close() throws IOException {
httpClient.close();
}
public void setAuthToken(String authToken) {
this.authToken = authToken;
}
private void readObject(java.io.ObjectInputStream in)
throws IOException, ClassNotFoundException {
in.defaultReadObject();
this.httpClient = authenticationProvider.createHttpClient();
}
public void httpGetBasicAuth(String path, OutputStream outputStream) throws HttpException {
logger.debug("HTTP GET W/ BASIC AUTH: " + path);
HttpGet request = new HttpGet(convertPath(path));
setBasicAuthHeader(request);
try (CloseableHttpResponse response = httpClient.execute(request)) {
verifyStatusCode(request, response);
if (response.getEntity() == null) {
throw new HttpException("Not body returned by url " + request.getURI());
}
IOUtils.copy(response.getEntity().getContent(), outputStream);
} catch (IOException e) {
throw new HttpException(e.getMessage(), e);
}
}
public String httpGet(String path, Environment env) throws HttpException {
logger.debug("HTTP GET " + path + " env=" + env);
return executeWithEnv(new HttpGet(convertPath(path)), env);
}
public String httpGet(String path) throws HttpException {
logger.debug("HTTP GET " + path);
return executeWrapper(new HttpGet(convertPath(path)), null);
}
public String httpGet(String path, Map headers) throws HttpException {
logger.debug("HTTP GET " + path);
HttpGet method = new HttpGet(convertPath(path));
setHeader(headers, method);
return executeWrapper(method, null);
}
public String httpGetWithOrgAndOwner(String path, String orgId, String ownerId) throws HttpException {
logger.debug("HTTP GET " + path);
return httpGet(path, createOrgAndOwnerHeaders(orgId, ownerId));
}
public String httpPost(String path, Object data, Environment env) throws HttpException {
logger.debug("HTTP POST " + path + " env=" + env + " data=" + data);
return executeWithDataAndEnv(new HttpPost(convertPath(path)), data, env);
}
public String httpPostWithOrgAndOwner(String path, Object data, String orgId, String ownerId) throws HttpException {
return httpPost(path, data, createOrgAndOwnerHeaders(orgId, ownerId));
}
public String httpPost(String path, Object data, Map headers) throws HttpException {
logger.debug("HTTP POST " + path + " headers=" + headers + " data=" + data);
HttpPost method = new HttpPost(convertPath(path));
setHeader(headers, method);
return execute(method, data);
}
public String httpPost(String path, Object data) throws HttpException {
logger.debug("HTTP POST " + path + " data=" + data);
return execute(new HttpPost(convertPath(path)), data);
}
public String httpPatch(String path, Object data) throws HttpException {
logger.debug("HTTP PATCH " + path + " data=" + data);
return execute(new HttpPatch(convertPath(path)), data);
}
public String httpPut(String path, Object data) throws HttpException {
logger.debug("HTTP PUT " + path + " data=" + data);
return execute(new HttpPut(convertPath(path)), data);
}
public String httpPut(String path, Object data, Environment environment) throws HttpException {
logger.debug("HTTP PUT " + path);
return executeWithDataAndEnv(new HttpPut(convertPath(path)), data, environment);
}
public String httpDelete(String path) throws HttpException {
logger.debug("HTTP DELETE " + path);
return executeWrapper(new HttpDelete(convertPath(path)), null);
}
public String httpDelete(String path, Object data) throws HttpException {
logger.debug("HTTP DELETE " + path + " data=" + data);
return execute(new HttpDeleteWithBody(convertPath(path)), data);
}
public String httpDelete(@NotNull String path, @NotNull Environment env) throws HttpException {
logger.debug("HTTP DELETE " + path + " env=" + env);
return executeWithEnv(new HttpDelete(convertPath(path)), env);
}
private void addEnvironmentHeaders(@Nullable Environment environment, HttpRequestBase request) {
if (environment != null) {
environment.addHeaders(request);
}
}
public MultiPartRequest createMultiPartPostRequest(@NotNull String url, @Nullable Environment environment) {
HttpPost request = new HttpPost(convertPath(url));
addEnvironmentHeaders(environment, request);
return new MultiPartRequest(request);
}
public MultiPartRequest createMultiPartPatchRequest(@NotNull String url, @Nullable Environment environment) {
HttpPatch request = new HttpPatch(convertPath(url));
addEnvironmentHeaders(environment, request);
return new MultiPartRequest(request);
}
public MultiPartRequest createMultiPartPutRequest(@NotNull String url, @Nullable Environment environment) {
HttpPut request = new HttpPut(convertPath(url));
addEnvironmentHeaders(environment, request);
return new MultiPartRequest(request);
}
private String execute(@NotNull HttpEntityEnclosingRequestBase method, Object data) throws HttpException {
if (data != null) {
if (data instanceof HttpEntity) {
method.setEntity((HttpEntity) data);
} else {
method.setHeader("Content-Type", "application/json");
method.setEntity(new ByteArrayEntity(client.getJsonHelper().toJson(data)));
}
}
return executeWrapper(method, null);
}
private String executeWithDataAndEnv(@NotNull HttpEntityEnclosingRequestBase method, @NotNull Object data, @NotNull Environment env) throws HttpException {
env.addHeaders(method);
return execute(method, data);
}
private String executeWithEnv(@NotNull HttpRequestBase method, @NotNull Environment env) throws HttpException {
env.addHeaders(method);
return executeWrapper(method, null);
}
protected String executeWrapper(@NotNull HttpRequestBase method, MultiPartRequest multiPartRequest) throws HttpException {
return executeWrapper(method, multiPartRequest, 0);
}
private String executeWrapper(@NotNull HttpRequestBase method, MultiPartRequest multiPartRequest, int attempt) throws HttpException {
if (isLoginRequest()) {
loginAttempts++;
}
try {
if (multiPartRequest != null) {
((HttpEntityEnclosingRequestBase) method).setEntity(multiPartRequest.toEntity());
}
return doExecute(method);
} catch (HttpException e) {
if (e.getStatusCode() == 403 || e.getStatusCode() == 401) {
if (loginAttempts > 1) {
throw e;
} else {
updateBearerToken();
return doExecute(method);
}
} else if (e.getStatusCode() >= 500) {
attempt++;
if (attempt > maxRetries) {
throw e;
} else {
ThreadUtils.sleep(retryDelay);
return executeWrapper(method, multiPartRequest, attempt);
}
} else {
throw e;
}
}
}
private void updateBearerToken() throws HttpException {
authToken = "bearer " + authenticationProvider.getBearerToken(this);
}
@Nullable
private synchronized String doExecute(HttpRequestBase method) throws HttpException {
if (authToken != null && method.getFirstHeader(HEADER_AUTH) == null) {
// if (authToken.startsWith("bearer ")) {
// authToken = "Bearer " + authToken.substring(7);
// }
method.setHeader(HEADER_AUTH, authToken);
}
try (CloseableHttpResponse response = httpClient.execute(method)) {
verifyStatusCode(method, response);
if (isLoginRequest()) {
loginAttempts = 0;
setLoginRequest(false);
}
if (response.getEntity() != null && response.getEntity().getContent() != null) {
String resStr = IOUtils.toString(response.getEntity().getContent());
logger.debug("RESULT CONTENT: " + resStr);
return resStr;
} else {
return null;
}
} catch (IOException e) {
throw new RuntimeIOException(e);
}
}
private void verifyStatusCode(HttpRequestBase method, CloseableHttpResponse response) throws IOException, HttpException {
int statusCode = response.getStatusLine().getStatusCode();
if (statusCode < 200 || statusCode > 299) {
String errMsg;
if (response.getEntity() != null && response.getEntity().getContent() != null) {
errMsg = " : " + IOUtils.toString(response.getEntity().getContent());
} else {
errMsg = "";
}
String message = "Anypoint returned status code " + statusCode + " - url: " + method.getURI() + " - err: " + errMsg;
if (statusCode == 403) {
throw new UnauthorizedHttpException(message, statusCode);
} else {
throw new HttpException(message, statusCode);
}
}
}
public synchronized void setProxy(@NotNull String scheme, @NotNull String host, int port,
@Nullable String username, @Nullable String password) {
if (httpClient != null) {
IOUtils.close(httpClient);
}
httpClient = authenticationProvider.createHttpClient(new HttpHost(host, port, scheme), username, password);
}
public synchronized void unsetProxy() {
if (httpClient != null) {
IOUtils.close(httpClient);
}
httpClient = authenticationProvider.createHttpClient();
}
public class MultiPartRequest {
private Map parts = new HashMap<>();
private HttpEntityEnclosingRequestBase request;
MultiPartRequest(HttpEntityEnclosingRequestBase request) {
this.request = request;
}
public MultiPartRequest addText(@NotNull String name, @NotNull String value) {
parts.put(name, value);
return this;
}
public MultiPartRequest addBinary(@NotNull String name, @NotNull StreamSource streamSource) {
parts.put(name, streamSource);
return this;
}
HttpEntity toEntity() throws HttpException {
try {
MultipartEntityBuilder builder = MultipartEntityBuilder.create();
for (Map.Entry e : parts.entrySet()) {
if (e.getValue() instanceof String) {
builder.addTextBody(e.getKey(), (String) e.getValue());
} else if (e.getValue() instanceof StreamSource) {
builder.addBinaryBody(e.getKey(), ((StreamSource) e.getValue()).createInputStream(),
ContentType.APPLICATION_OCTET_STREAM, ((StreamSource) e.getValue()).getFileName());
}
}
return builder.build();
} catch (IOException e) {
throw new HttpException("Failed to read data to send: " + e.getMessage(), e);
}
}
public String execute() throws HttpException, IOException {
try {
logger.debug("HTTP {}", request);
return HttpHelper.this.executeWrapper(request, this);
} catch (RuntimeIOException e) {
throw e.getIOException();
}
}
}
public int getMaxRetries() {
return maxRetries;
}
public void setMaxRetries(int maxRetries) {
this.maxRetries = maxRetries;
}
public long getRetryDelay() {
return retryDelay;
}
public void setRetryDelay(long retryDelay) {
this.retryDelay = retryDelay;
}
public boolean isLoginRequest() {
return loginRequest;
}
public void setLoginRequest(boolean loginRequest) {
this.loginRequest = loginRequest;
}
public class RuntimeIOException extends RuntimeException {
RuntimeIOException(@NotNull IOException ioException) {
super(ioException.getMessage(), ioException);
}
@NotNull
IOException getIOException() {
return (IOException) getCause();
}
}
private static void setHeader(Map headers, HttpRequestBase method) {
for (Map.Entry entry : headers.entrySet()) {
method.setHeader(entry.getKey(), entry.getValue());
}
}
public void setClient(AnypointClient client) {
this.client = client;
}
protected static String convertPath(String path) {
return path.startsWith("/") ? "https://anypoint.mulesoft.com" + path : path;
}
@NotNull
private static HashMap createOrgAndOwnerHeaders(String orgId, String ownerId) {
HashMap headers = new HashMap<>();
headers.put("x-organization-id", orgId);
headers.put("x-owner-id", ownerId);
return headers;
}
private HttpRequestBase setBasicAuthHeader(HttpRequestBase request) {
if (authenticationProvider instanceof AuthenticationProviderUsernamePasswordImpl) {
String authStr = ((AuthenticationProviderUsernamePasswordImpl) authenticationProvider).getUsername() +
":" + ((AuthenticationProviderUsernamePasswordImpl) authenticationProvider).getPassword();
byte[] encodedAuth = Base64.encodeBase64(authStr.getBytes(StandardCharsets.ISO_8859_1));
String authHeader = "Basic " + new String(encodedAuth);
request.setHeader(HttpHeaders.AUTHORIZATION, authHeader);
return request;
} else {
throw new RuntimeException("Only supported with username/password authentication at this time");
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy