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

com.rabbitmq.http.client.JdkHttpClientHttpLayer Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2021 the original author or authors.
 *
 * 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
 *
 *      https://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.rabbitmq.http.client;

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Type;
import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpClient.Builder;
import java.net.http.HttpClient.Redirect;
import java.net.http.HttpClient.Version;
import java.net.http.HttpRequest;
import java.net.http.HttpRequest.BodyPublishers;
import java.net.http.HttpResponse;
import java.net.http.HttpResponse.BodyHandlers;
import java.time.Duration;
import java.util.Collections;
import java.util.Map;
import java.util.function.Consumer;
import java.util.function.Supplier;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLParameters;

/**
 * {@link HttpLayer} using JDK 11's {@link HttpClient}.
 *
 * @since 4.0.0
 */
public final class JdkHttpClientHttpLayer implements HttpLayer {

  private static final Duration REQUEST_TIMEOUT = Duration.ofSeconds(60);
  private static final Duration CONNECT_TIMEOUT = Duration.ofSeconds(30);

  private final HttpClient client;
  private final ObjectMapper mapper;
  private final Consumer requestBuilderConsumer;

  private JdkHttpClientHttpLayer(
      HttpClient client,
      ObjectMapper mapper,
      Consumer requestBuilderConsumer) {
    this.client = client;
    this.mapper = mapper;
    this.requestBuilderConsumer = requestBuilderConsumer;
  }

  static  T get(
      URI uri,
      ObjectMapper mapper,
      HttpClient client,
      Consumer requestBuilderConsumer,
      Type type) {
    HttpRequest.Builder requestBuilder = HttpRequest.newBuilder(uri);
    requestBuilder.header("accept", "application/json");
    requestBuilderConsumer.accept(requestBuilder);
    HttpRequest request = requestBuilder.GET().build();
    try {
      HttpResponse> response =
          client.send(request, new JsonBodyHandler<>(mapper, type));
      int statusCode = response.statusCode();
      if (statusCode == 404) {
        return null;
      } else if (statusCode == 200) {
        return response.body().get();
      } else {
        int errorClass = errorClass(statusCode);
        if (errorClass == 400) {
          throw new HttpClientException(statusCode, "GET returned " + statusCode);
        } else if (errorClass == 500) {
          throw new HttpServerException(statusCode, "GET returned " + statusCode);
        } else {
          throw new HttpException("GET returned " + statusCode);
        }
      }
    } catch (IOException e) {
      throw new HttpException(e);
    } catch (InterruptedException e) {
      Thread.currentThread().interrupt();
      throw new HttpException(e);
    }
  }

  private static  java.net.http.HttpResponse.BodySubscriber> asJson(
      ObjectMapper mapper, Type type) {
    java.net.http.HttpResponse.BodySubscriber upstream =
        java.net.http.HttpResponse.BodySubscribers.ofInputStream();

    return java.net.http.HttpResponse.BodySubscribers.mapping(
        upstream, inputStream -> toSupplierOfType(mapper, inputStream, type));
  }

  private static  Supplier toSupplierOfType(
      ObjectMapper mapper, InputStream inputStream, Type type) {
    return () -> {
      try (InputStream stream = inputStream) {
        return mapper.readValue(stream, mapper.constructType(type));
      } catch (IOException e) {
        throw new HttpException(e);
      }
    };
  }

  static void maybeThrowClientServerException(int statusCode, String message, int... ignores) {
    for (int ignore : ignores) {
      if (statusCode == ignore) {
        return;
      }
    }
    int errorClass = errorClass(statusCode);
    if (errorClass == 400) {
      throw new HttpClientException(statusCode, message);
    } else if (errorClass == 500) {
      throw new HttpServerException(statusCode, message);
    }
  }

  private static int errorClass(int statusCode) {
    return statusCode - statusCode % 100;
  }

  /**
   * Configure the default {@link HttpLayer} implementation.
   *
   * @return a configuration instance
   */
  public static Configuration configure() {
    return new Configuration();
  }

  public static String authorization(String username, String password) {
    return "Basic " + Utils.base64(username + ":" + password);
  }

  private static HttpRequest.BodyPublisher bodyPublisher(ObjectMapper mapper, Object requestBody) throws JsonProcessingException {
    return requestBody == null ? BodyPublishers.noBody() :
        BodyPublishers.ofByteArray(mapper.writeValueAsBytes(requestBody));
  }

  @Override
  public  T get(URI uri, Class responseClass) {
    return get(uri, this.mapper, this.client, this.requestBuilderConsumer, responseClass);
  }

  @Override
  public  T get(URI uri, ParameterizedTypeReference typeReference) {
    return get(uri, this.mapper, this.client, this.requestBuilderConsumer, typeReference.getType());
  }

  @Override
  public  T post(URI uri, Object requestBody, Class responseClass) {
    HttpRequest.Builder requestBuilder = HttpRequest.newBuilder(uri);
    requestBuilder.header("content-type", "application/json");
    requestBuilderConsumer.accept(requestBuilder);
    try {
      HttpRequest request =
          requestBuilder
              .POST(bodyPublisher(mapper, requestBody))
              .build();
      if (responseClass == null) {
        HttpResponse response = client.send(request, BodyHandlers.discarding());
        int statusCode = response.statusCode();
        maybeThrowClientServerException(statusCode, "POST returned " + statusCode);
        return null;
      } else {
        HttpResponse> response =
            client.send(request, new JsonBodyHandler<>(mapper, responseClass));
        int statusCode = response.statusCode();
        maybeThrowClientServerException(statusCode, "POST returned " + statusCode);
        return response.body().get();
      }
    } catch (IOException e) {
      throw new HttpException(e);
    } catch (InterruptedException e) {
      Thread.currentThread().interrupt();
      throw new HttpException(e);
    }
  }

  @Override
  public void put(URI uri, Object requestBody) {
    HttpRequest.Builder requestBuilder = HttpRequest.newBuilder(uri);
    requestBuilder.header("content-type", "application/json");
    requestBuilderConsumer.accept(requestBuilder);
    try {
      HttpRequest request =
          requestBuilder
              .PUT(bodyPublisher(mapper, requestBody))
              .build();
      HttpResponse response = client.send(request, BodyHandlers.discarding());
      int statusCode = response.statusCode();
      maybeThrowClientServerException(statusCode, "PUT returned " + statusCode);
    } catch (IOException e) {
      throw new HttpException(e);
    } catch (InterruptedException e) {
      Thread.currentThread().interrupt();
      throw new HttpException(e);
    }
  }

  @Override
  public void delete(URI uri, Map headers) {
    headers = headers == null ? Collections.emptyMap() : headers;
    HttpRequest.Builder requestBuilder = HttpRequest.newBuilder(uri);
    headers.forEach(requestBuilder::header);
    requestBuilderConsumer.accept(requestBuilder);
    try {
      HttpRequest request = requestBuilder.DELETE().build();
      HttpResponse response = client.send(request, BodyHandlers.discarding());
      int statusCode = response.statusCode();
      maybeThrowClientServerException(statusCode, "DELETE returned " + statusCode, 404);
    } catch (IOException e) {
      throw new HttpException(e);
    } catch (InterruptedException e) {
      Thread.currentThread().interrupt();
      throw new HttpException(e);
    }
  }

  private static class Factory implements HttpLayerFactory {

    private final Consumer clientBuilderConsumer;
    private final Consumer requestBuilderConsumer;

    Factory(
        Consumer clientBuilderConsumer,
        Consumer requestBuilderConsumer) {
      this.clientBuilderConsumer = clientBuilderConsumer;
      this.requestBuilderConsumer = requestBuilderConsumer;
    }

    @Override
    public HttpLayer create(ClientParameters parameters) {
      HttpClient.Builder builder =
          HttpClient.newBuilder()
              .version(Version.HTTP_1_1)
              .followRedirects(Redirect.NORMAL)
              .connectTimeout(CONNECT_TIMEOUT);
      this.clientBuilderConsumer.accept(builder);
      HttpClient client = builder.build();
      ObjectMapper mapper = JsonUtils.createDefaultObjectMapper();
      String username = parameters.getUsername();
      String password = parameters.getPassword();
      Consumer requestBuilderConsumer;
      if (this.requestBuilderConsumer == null) {
        requestBuilderConsumer =
            requestBuilder ->
                requestBuilder
                    .timeout(REQUEST_TIMEOUT)
                    .setHeader("Authorization", authorization(username, password));
      } else {
        requestBuilderConsumer = this.requestBuilderConsumer;
      }
      return new JdkHttpClientHttpLayer(client, mapper, requestBuilderConsumer);
    }
  }

  private static class JsonBodyHandler
      implements java.net.http.HttpResponse.BodyHandler> {

    private final ObjectMapper mapper;
    private final Type type;

    public JsonBodyHandler(ObjectMapper mapper, Type type) {
      this.mapper = mapper;
      this.type = type;
    }

    @Override
    public java.net.http.HttpResponse.BodySubscriber> apply(
        HttpResponse.ResponseInfo responseInfo) {
      return asJson(mapper, type);
    }
  }

  /**
   * Class to configure the default {@link HttpLayer} implementation.
   *
   * @see JdkHttpClientHttpLayer
   */
  public static class Configuration {

    private Consumer clientBuilderConsumer = b -> {};
    private Consumer requestBuilderConsumer = null;

    /**
     * Callback to configure the {@link Builder}.
     *
     * 

The client can be configured to use TLS with this callback. Use the {@link * Builder#sslContext(SSLContext)} and {@link Builder#sslParameters(SSLParameters)} methods to * configure TLS appropriately. * *

Use the {@link SSLParameters#setEndpointIdentificationAlgorithm(String)} method with the * HTTPS value to enable server hostname verification. * * @param clientBuilderConsumer * @return this configuration instance */ public Configuration clientBuilderConsumer(Consumer clientBuilderConsumer) { if (clientBuilderConsumer == null) { throw new IllegalArgumentException("Client builder consumer cannot be null"); } this.clientBuilderConsumer = clientBuilderConsumer; return this; } /** * Callback to configure the {@link HttpRequest.Builder}. * * @param requestBuilderConsumer * @return the configuration instance */ public Configuration requestBuilderConsumer( Consumer requestBuilderConsumer) { if (requestBuilderConsumer == null) { throw new IllegalArgumentException("Request builder consumer cannot be null"); } this.requestBuilderConsumer = requestBuilderConsumer; return this; } public HttpLayerFactory create() { return new Factory(this.clientBuilderConsumer, this.requestBuilderConsumer); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy