OkHttpBuilder.java

  1. /**
  2.  * Copyright (C) 2017 HttpBuilder-NG Project
  3.  *
  4.  * Licensed under the Apache License, Version 2.0 (the "License");
  5.  * you may not use this file except in compliance with the License.
  6.  * You may obtain a copy of the License at
  7.  *
  8.  *  http://www.apache.org/licenses/LICENSE-2.0
  9.  *
  10.  * Unless required by applicable law or agreed to in writing, software
  11.  * distributed under the License is distributed on an "AS IS" BASIS,
  12.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13.  * See the License for the specific language governing permissions and
  14.  * limitations under the License.
  15.  */
  16. package groovyx.net.http;

  17. import com.burgstaller.okhttp.AuthenticationCacheInterceptor;
  18. import com.burgstaller.okhttp.CachingAuthenticatorDecorator;
  19. import com.burgstaller.okhttp.digest.CachingAuthenticator;
  20. import com.burgstaller.okhttp.digest.DigestAuthenticator;
  21. import groovy.lang.Closure;
  22. import groovy.lang.DelegatesTo;
  23. import groovyx.net.http.util.IoUtils;
  24. import okhttp3.*;
  25. import okio.BufferedSink;

  26. import javax.net.ssl.SSLContext;
  27. import java.io.IOException;
  28. import java.io.InputStream;
  29. import java.net.Proxy;
  30. import java.net.URI;
  31. import java.net.URISyntaxException;
  32. import java.nio.charset.Charset;
  33. import java.util.ArrayList;
  34. import java.util.LinkedHashMap;
  35. import java.util.List;
  36. import java.util.Map;
  37. import java.util.concurrent.ConcurrentHashMap;
  38. import java.util.concurrent.Executor;
  39. import java.util.function.Consumer;
  40. import java.util.function.Function;

  41. import static groovyx.net.http.FromServer.Header.keyValue;
  42. import static groovyx.net.http.HttpBuilder.ResponseHandlerFunction.HANDLER_FUNCTION;
  43. import static groovyx.net.http.HttpConfig.AuthType.DIGEST;
  44. import static okhttp3.MediaType.parse;

  45. /**
  46.  * `HttpBuilder` implementation based on the http://square.github.io/okhttp/[OkHttp] client library.
  47.  *
  48.  * Generally, this class should not be used directly, the preferred method of instantiation is via one of the two static `configure()` methods of this
  49.  * class or using one of the `configure` methods of `HttpBuilder` with a factory function for this builder.
  50.  */
  51. public class OkHttpBuilder extends HttpBuilder {

  52.     private static final Function<HttpObjectConfig, ? extends HttpBuilder> okFactory = OkHttpBuilder::new;
  53.     private static final String OPTIONS = "OPTIONS";
  54.     private static final String TRACE = "TRACE";
  55.     private final ChainedHttpConfig config;
  56.     private final HttpObjectConfig.Client clientConfig;
  57.     private final Executor executor;
  58.     private final OkHttpClient client;

  59.     protected OkHttpBuilder(final HttpObjectConfig config) {
  60.         super(config);

  61.         this.config = config.getChainedConfig();
  62.         this.clientConfig = config.getClient();
  63.         this.executor = config.getExecution().getExecutor();

  64.         final OkHttpClient.Builder builder = new OkHttpClient.Builder();

  65.         final SSLContext sslContext = config.getExecution().getSslContext();
  66.         if (sslContext != null) {
  67.             builder.sslSocketFactory(sslContext.getSocketFactory()/*, (X509TrustManager) TRUST_MANAGERS[0]*/);
  68.             builder.hostnameVerifier(config.getExecution().getHostnameVerifier());
  69.         }

  70.         // DIGEST support - defining this here only allows DIGEST config on the HttpBuilder configuration, not for individual methods.
  71.         final HttpConfig.Auth auth = config.getRequest().getAuth();
  72.         if (auth != null && auth.getAuthType() == DIGEST) {
  73.             Map<String, CachingAuthenticator> authCache = new ConcurrentHashMap<>();

  74.             builder.addInterceptor(new AuthenticationCacheInterceptor(authCache));
  75.             builder.authenticator(new CachingAuthenticatorDecorator(
  76.                 new DigestAuthenticator(new com.burgstaller.okhttp.digest.Credentials(auth.getUser(), auth.getPassword())), authCache)
  77.             );
  78.         }

  79.         final Consumer<Object> clientCustomizer = clientConfig.getClientCustomizer();
  80.         if (clientCustomizer != null) {
  81.             clientCustomizer.accept(builder);
  82.         }

  83.         final ProxyInfo pinfo = config.getExecution().getProxyInfo();
  84.         if (usesProxy(pinfo)) {
  85.             builder.proxy(pinfo.getProxy());
  86.         }

  87.         this.client = builder.build();
  88.     }

  89.     private boolean usesProxy(final ProxyInfo pinfo) {
  90.         return pinfo != null && pinfo.getProxy().type() != Proxy.Type.DIRECT;
  91.     }

  92.     /**
  93.      * Retrieves the internal client implementation as an {@link OkHttpClient} instance.
  94.      *
  95.      * @return the reference to the internal client implementation as an {@link OkHttpClient}
  96.      */
  97.     public Object getClientImplementation() {
  98.         return client;
  99.     }

  100.     /**
  101.      * Creates an `HttpBuilder` using the `OkHttpBuilder` factory instance configured with the provided configuration closure.
  102.      *
  103.      * The configuration closure delegates to the {@link HttpObjectConfig} interface, which is an extension of the {@link HttpConfig} interface -
  104.      * configuration properties from either may be applied to the global client configuration here. See the documentation for those interfaces for
  105.      * configuration property details.
  106.      *
  107.      * [source,groovy]
  108.      * ----
  109.      * def http = HttpBuilder.configure {
  110.      *     request.uri = 'http://localhost:10101'
  111.      * }
  112.      * ----
  113.      *
  114.      * @param closure the configuration closure (delegated to {@link HttpObjectConfig})
  115.      * @return the configured `HttpBuilder`
  116.      */
  117.     public static HttpBuilder configure(@DelegatesTo(HttpObjectConfig.class) final Closure closure) {
  118.         return configure(okFactory, closure);
  119.     }

  120.     /**
  121.      * Creates an `HttpBuilder` using the `OkHttpBuilder` factory instance configured with the provided configuration function.
  122.      *
  123.      * The configuration {@link Consumer} function accepts an instance of the {@link HttpObjectConfig} interface, which is an extension of the {@link HttpConfig}
  124.      * interface - configuration properties from either may be applied to the global client configuration here. See the documentation for those interfaces for
  125.      * configuration property details.
  126.      *
  127.      * This configuration method is generally meant for use with standard Java.
  128.      *
  129.      * [source,java]
  130.      * ----
  131.      * HttpBuilder.configure(new Consumer<HttpObjectConfig>() {
  132.      * public void accept(HttpObjectConfig config) {
  133.      *     config.getRequest().setUri(format("http://localhost:%d", serverRule.getPort()));
  134.      * }
  135.      * });
  136.      * ----
  137.      *
  138.      * Or, using lambda expressions:
  139.      *
  140.      * [source,java]
  141.      * ----
  142.      * HttpBuilder.configure(config -> {
  143.      *     config.getRequest().setUri(format("http://localhost:%d", serverRule.getPort()));
  144.      * });
  145.      * ----
  146.      *
  147.      * @param configuration the configuration function (accepting {@link HttpObjectConfig})
  148.      * @return the configured `HttpBuilder`
  149.      */
  150.     public static HttpBuilder configure(final Consumer<HttpObjectConfig> configuration) {
  151.         return configure(okFactory, configuration);
  152.     }

  153.     @Override
  154.     protected ChainedHttpConfig getObjectConfig() {
  155.         return config;
  156.     }

  157.     @Override
  158.     public Executor getExecutor() {
  159.         return executor;
  160.     }

  161.     @Override
  162.     protected Object doGet(final ChainedHttpConfig chainedConfig) {
  163.         return execute((url) -> new Request.Builder().get().url(url), chainedConfig);
  164.     }

  165.     @Override
  166.     protected Object doHead(final ChainedHttpConfig chainedConfig) {
  167.         return execute((url) -> new Request.Builder().head().url(url), chainedConfig);
  168.     }

  169.     @Override
  170.     protected Object doPost(final ChainedHttpConfig chainedConfig) {
  171.         return execute((url) -> new Request.Builder().post(resolveRequestBody(chainedConfig)).url(url), chainedConfig);
  172.     }

  173.     @Override
  174.     protected Object doPut(final ChainedHttpConfig chainedConfig) {
  175.         return execute((url) -> new Request.Builder().put(resolveRequestBody(chainedConfig)).url(url), chainedConfig);
  176.     }

  177.     @Override
  178.     protected Object doPatch(final ChainedHttpConfig chainedConfig) {
  179.         return execute((url) -> new Request.Builder().patch(resolveRequestBody(chainedConfig)).url(url), chainedConfig);
  180.     }

  181.     @Override
  182.     protected Object doDelete(final ChainedHttpConfig chainedConfig) {
  183.         return execute((url) -> new Request.Builder().delete().url(url), chainedConfig);
  184.     }

  185.     @Override
  186.     protected Object doOptions(final ChainedHttpConfig config) {
  187.         return execute((url) -> new Request.Builder().method(OPTIONS, null).url(url), config);
  188.     }

  189.     @Override
  190.     protected Object doTrace(final ChainedHttpConfig config) {
  191.         return execute((url) -> new Request.Builder().method(TRACE, null).url(url), config);
  192.     }

  193.     @Override
  194.     public void close() throws IOException {
  195.         // does nothing
  196.     }

  197.     private RequestBody resolveRequestBody(final ChainedHttpConfig chainedConfig) {
  198.         final ChainedHttpConfig.ChainedRequest cr = chainedConfig.getChainedRequest();

  199.         final RequestBody body;
  200.         if (cr.actualBody() != null) {
  201.             final OkHttpToServer toServer = new OkHttpToServer(chainedConfig);
  202.             chainedConfig.findEncoder().accept(chainedConfig, toServer);
  203.             body = toServer;

  204.         } else {
  205.             body = RequestBody.create(resolveMediaType(cr.actualContentType(), cr.actualCharset()), "");
  206.         }

  207.         return body;
  208.     }

  209.     private static MediaType resolveMediaType(final String contentType, final Charset charset) {
  210.         if (contentType != null) {
  211.             if (charset != null) {
  212.                 return parse(contentType + "; charset=" + charset.toString().toLowerCase());
  213.             } else {
  214.                 return parse(contentType);
  215.             }
  216.         }
  217.         return null;
  218.     }

  219.     @SuppressWarnings("Duplicates")
  220.     private void applyHeaders(final Request.Builder requestBuilder, final ChainedHttpConfig.ChainedRequest cr) throws URISyntaxException {
  221.         for (Map.Entry<String, CharSequence> entry : cr.actualHeaders(new LinkedHashMap<>()).entrySet()) {
  222.             requestBuilder.addHeader(entry.getKey(), entry.getValue() != null ? entry.getValue().toString() : null);
  223.         }

  224.         for (Map.Entry<String, String> e : cookiesToAdd(clientConfig, cr).entrySet()) {
  225.             requestBuilder.addHeader(e.getKey(), e.getValue());
  226.         }
  227.     }

  228.     private static void applyAuth(final Request.Builder requestBuilder, final ChainedHttpConfig chainedConfig) {
  229.         final HttpConfig.Auth auth = chainedConfig.getChainedRequest().actualAuth();
  230.         if (auth != null) {
  231.             switch (auth.getAuthType()) {
  232.                 case BASIC:
  233.                     requestBuilder.addHeader("Authorization", Credentials.basic(auth.getUser(), auth.getPassword()));
  234.                     break;
  235.                 case DIGEST:
  236.                     // supported in constructor with an interceptor
  237.             }
  238.         }
  239.     }

  240.     private Object execute(final Function<HttpUrl, Request.Builder> makeBuilder, final ChainedHttpConfig chainedConfig) {
  241.         try {
  242.             final ChainedHttpConfig.ChainedRequest cr = chainedConfig.getChainedRequest();
  243.             final URI uri = cr.getUri().toURI();
  244.             final HttpUrl httpUrl = HttpUrl.get(uri);
  245.             final Request.Builder requestBuilder = makeBuilder.apply(httpUrl);

  246.             applyHeaders(requestBuilder, cr);

  247.             applyAuth(requestBuilder, chainedConfig);

  248.             try (Response response = client.newCall(requestBuilder.build()).execute()) {
  249.                 return HANDLER_FUNCTION.apply(chainedConfig, new OkHttpFromServer(chainedConfig.getChainedRequest().getUri().toURI(), response));
  250.             } catch (IOException ioe) {
  251.                 throw ioe; //re-throw, close has happened
  252.             }
  253.         } catch (Exception e) {
  254.             return handleException(chainedConfig.getChainedResponse(), e);
  255.         }
  256.     }

  257.     private class OkHttpFromServer implements FromServer {

  258.         private final URI uri;
  259.         private final Response response;
  260.         private List<Header<?>> headers;
  261.         private boolean body;

  262.         private OkHttpFromServer(final URI uri, final Response response) {
  263.             this.uri = uri;
  264.             this.response = response;
  265.             this.headers = populateHeaders();

  266.             addCookieStore(uri, headers);

  267.             try {
  268.                 body = !response.body().source().exhausted() && response.peekBody(1).bytes().length > 0;
  269.             } catch (IOException e) {
  270.                 body = false;
  271.             }
  272.         }

  273.         private List<Header<?>> populateHeaders() {
  274.             final Headers headers = response.headers();
  275.             List<Header<?>> ret = new ArrayList<>();
  276.             for (String name : headers.names()) {
  277.                 List<String> values = headers.values(name);
  278.                 for (String value : values) {
  279.                     ret.add(keyValue(name, value));
  280.                 }
  281.             }

  282.             return ret;
  283.         }

  284.         @Override
  285.         public InputStream getInputStream() {
  286.             return response.body().byteStream();
  287.         }

  288.         @Override
  289.         public int getStatusCode() {
  290.             return response.code();
  291.         }

  292.         @Override
  293.         public String getMessage() {
  294.             return response.message();
  295.         }

  296.         @Override
  297.         public List<Header<?>> getHeaders() {
  298.             return headers;
  299.         }

  300.         @Override
  301.         public boolean getHasBody() {
  302.             return body;
  303.         }

  304.         @Override
  305.         public URI getUri() {
  306.             return uri;
  307.         }

  308.         @Override
  309.         public void finish() {
  310.             response.close();
  311.         }
  312.     }

  313.     private static class OkHttpToServer extends RequestBody implements ToServer {

  314.         private ChainedHttpConfig config;
  315.         private byte[] bytes;

  316.         private OkHttpToServer(final ChainedHttpConfig config) {
  317.             this.config = config;
  318.         }

  319.         @Override
  320.         public void toServer(final InputStream inputStream) {
  321.             try {
  322.                 this.bytes = IoUtils.streamToBytes(inputStream);
  323.             } catch (IOException e) {
  324.                 throw new RuntimeException(e);
  325.             }
  326.         }

  327.         @Override
  328.         public MediaType contentType() {
  329.             return resolveMediaType(config.findContentType(), config.findCharset());
  330.         }

  331.         @Override
  332.         public long contentLength() throws IOException {
  333.             return bytes.length;
  334.         }

  335.         @Override
  336.         public void writeTo(final BufferedSink sink) throws IOException {
  337.             sink.write(bytes);
  338.         }
  339.     }
  340. }