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

com.databricks.sdk.core.http.Request Maven / Gradle / Ivy

The newest version!
package com.databricks.sdk.core.http;

import com.databricks.sdk.core.DatabricksException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.net.URI;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.*;

public class Request {
  public static final String GET = "GET";
  public static final String HEAD = "HEAD";
  public static final String DELETE = "DELETE";
  public static final String POST = "POST";
  public static final String PUT = "PUT";
  public static final String PATCH = "PATCH";
  private final String method;
  private String url;
  private final Map headers = new HashMap<>();
  private final Map> query = new TreeMap<>();
  private Optional redirectionBehavior = Optional.empty();
  /**
   * The body of the request for requests with streaming bodies. At most one of {@link #bodyStream}
   * and {@link #bodyString} can be non-null.
   */
  private final InputStream bodyStream;
  /**
   * The body of the request for requests with string bodies. At most one of {@link #bodyStream} and
   * {@link #bodyString} can be non-null.
   */
  private final String bodyString;
  /**
   * Whether the body of the request is a streaming body. At most one of {@link #isBodyStreaming}
   * and {@link #isBodyString} can be true.
   */
  private final boolean isBodyStreaming;
  /**
   * Whether the body of the request is a string body. At most one of {@link #isBodyStreaming} and
   * {@link #isBodyString} can be true.
   */
  private final boolean isBodyString;

  public Request(String method, String url) {
    this(method, url, null, null);
  }

  public Request(String method, String url, String bodyString) {
    this(method, url, null, bodyString);
  }

  public Request(String method, String url, InputStream bodyStream) {
    this(method, url, bodyStream, null);
  }

  private Request(String method, String url, InputStream bodyStream, String bodyString) {
    if (bodyStream != null && bodyString != null) {
      throw new IllegalArgumentException(
          "At most one of bodyStream and bodyString can be non-null");
    }
    this.method = method;
    this.url = url;
    this.bodyStream = bodyStream;
    this.bodyString = bodyString;
    this.isBodyStreaming = bodyStream != null;
    this.isBodyString = bodyString != null;
  }

  public Request withHeaders(Map headers) {
    this.headers.putAll(headers);
    return this;
  }

  public Request withHeader(String key, String value) {
    headers.put(key, value);
    return this;
  }

  public Request withQueryParam(String key, String value) {
    List values = query.computeIfAbsent(key, k -> new ArrayList<>());
    values.add(value);
    return this;
  }

  public Request withUrl(String url) {
    this.url = url;
    return this;
  }

  protected static String mapToQuery(Map> in) {
    StringJoiner joiner = new StringJoiner("&");
    for (Map.Entry> entry : in.entrySet()) {
      try {
        String encodedKey = URLEncoder.encode(entry.getKey(), StandardCharsets.UTF_8.name());
        for (String value : entry.getValue()) {
          String encodedValue = URLEncoder.encode(value, StandardCharsets.UTF_8.name());
          joiner.add(encodedKey + "=" + encodedValue);
        }
      } catch (UnsupportedEncodingException e) {
        throw new DatabricksException("Unable to encode query parameter: " + e.getMessage(), e);
      }
    }
    return joiner.toString();
  }

  public URI getUri() {
    URI uri = URI.create(url);
    if (!query.isEmpty()) {
      String rawQuery = uri.getRawQuery();
      rawQuery = rawQuery == null ? "" : rawQuery + "&";
      rawQuery += mapToQuery(query);
      // We need to construct the query by hand and cannot use the new URI(uri.getScheme(),
      // uri.getAuthority(),
      // uri.getPath(), rawQuery, uri.getFragment()) constructor. This constructor does not
      // percent-encode certain
      // characters in the query, such as '/'. Instead, the query is encoded here, and the complete
      // URI is passed into
      // the new URI(String) constructor, which assumes that the URI is already encoded.
      StringBuilder updatedUriString = new StringBuilder();
      if (uri.getScheme() != null) {
        updatedUriString.append(uri.getScheme()).append("://");
      }
      if (uri.getAuthority() != null) {
        updatedUriString.append(uri.getAuthority());
      }
      updatedUriString.append(uri.getPath());
      if (!rawQuery.isEmpty()) {
        updatedUriString.append("?").append(rawQuery);
      }
      if (uri.getFragment() != null) {
        updatedUriString.append("#").append(uri.getFragment());
      }
      return URI.create(updatedUriString.toString());
    }
    return uri;
  }

  public String getRequestLine() {
    URI uri = getUri();
    String path = uri.getPath();
    if (!query.isEmpty()) {
      path += "?" + uri.getQuery();
    }
    return String.format("%s %s", method, path);
  }

  public String getMethod() {
    return method;
  }

  public String getUrl() {
    return url;
  }

  public Map getHeaders() {
    return headers;
  }

  public Map> getQuery() {
    return query;
  }

  public InputStream getBodyStream() {
    return bodyStream;
  }

  public String getBodyString() {
    return bodyString;
  }

  public boolean isBodyStreaming() {
    return isBodyStreaming;
  }

  public boolean isBodyString() {
    return isBodyString;
  }

  @Override
  public boolean equals(Object o) {
    if (this == o) return true;
    if (o == null || getClass() != o.getClass()) return false;
    Request request = (Request) o;
    return method.equals(request.method)
        && url.equals(request.url)
        && Objects.equals(query, request.query)
        && Objects.equals(bodyStream, request.bodyStream)
        && Objects.equals(redirectionBehavior, request.redirectionBehavior);
  }

  @Override
  public int hashCode() {
    // Note: this is not safe for production, as bodyString will be null for different requests
    // using InputStream.
    // It is currently only used in tests.
    return Objects.hash(method, url, query, bodyString);
  }

  @Override
  public String toString() {
    return getRequestLine();
  }

  public Optional getRedirectionBehavior() {
    return this.redirectionBehavior;
  }

  public void setRedirectionBehavior(boolean redirectionBehavior) {
    this.redirectionBehavior = Optional.of(redirectionBehavior);
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy