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

sttp.client3.RequestT.scala Maven / Gradle / Ivy

There is a newer version: 3.10.1
Show newest version
package sttp.client3

import sttp.capabilities.{Effect, Streams}
import sttp.client3.internal.DigestAuthenticator.DigestAuthData
import sttp.client3.internal.{SttpFile, ToCurlConverter, ToRfc2616Converter, _}
import sttp.client3.logging.LoggingOptions
import sttp.model._
import sttp.model.headers.CookieWithMeta

import java.io.InputStream
import java.nio.ByteBuffer
import java.util.Base64
import scala.collection.immutable.Seq
import scala.concurrent.duration.Duration

/** Describes a HTTP request, along with a description of how the response body should be handled.
  *
  * The request can be sent:
  *
  *   - synchronously, using [[SimpleHttpClient.send()]]
  *   - using the [[send(SttpBackend)]] methods, which support any effect. The backend must provide a superset of the
  *     capabilities required by the request.
  *
  * @param response
  *   Description of how the response body should be handled. Needs to be specified upfront so that the response is
  *   always consumed and hence there are no requirements on client code to consume it. An exception to this are unsafe
  *   streaming and websocket responses, which need to be consumed/closed by the client.
  * @param tags
  *   Request-specific tags which can be used by backends for logging, metrics, etc. Not used by default.
  * @tparam U
  *   Specifies if the method & uri are specified. By default can be either: * [[Empty]], which is a type constructor
  *   which always resolves to [[None]]. This type of request is aliased to [[PartialRequest]]: there's no method and
  *   uri specified, and the request cannot be sent. * [[Identity]], which is an identity type constructor. This type of
  *   request is aliased to [[Request]]: the method and uri are specified, and the request can be sent.
  * @tparam T
  *   The target type, to which the response body should be read.
  * @tparam R
  *   The backend capabilities required by the request or response description. This might be `Any` (no requirements),
  *   [[Effect]] (the backend must support the given effect type), [[Streams]] (the ability to send and receive
  *   streaming bodies) or [[sttp.capabilities.WebSockets]] (the ability to handle websocket requests).
  */
case class RequestT[U[_], T, -R](
    method: U[Method],
    uri: U[Uri],
    body: RequestBody[R],
    headers: Seq[Header],
    response: ResponseAs[T, R],
    options: RequestOptions,
    tags: Map[String, Any]
) extends HasHeaders
    with RequestTExtensions[U, T, R] {
  def get(uri: Uri): Request[T, R] =
    this.copy[Identity, T, R](uri = uri, method = Method.GET)
  def head(uri: Uri): Request[T, R] =
    this.copy[Identity, T, R](uri = uri, method = Method.HEAD)
  def post(uri: Uri): Request[T, R] =
    this.copy[Identity, T, R](uri = uri, method = Method.POST)
  def put(uri: Uri): Request[T, R] =
    this.copy[Identity, T, R](uri = uri, method = Method.PUT)
  def delete(uri: Uri): Request[T, R] =
    this.copy[Identity, T, R](uri = uri, method = Method.DELETE)
  def options(uri: Uri): Request[T, R] =
    this.copy[Identity, T, R](uri = uri, method = Method.OPTIONS)
  def patch(uri: Uri): Request[T, R] =
    this.copy[Identity, T, R](uri = uri, method = Method.PATCH)
  def method(method: Method, uri: Uri): Request[T, R] =
    this.copy[Identity, T, R](uri = uri, method = method)

  def contentType(ct: String): RequestT[U, T, R] =
    header(HeaderNames.ContentType, ct, replaceExisting = true)
  def contentType(mt: MediaType): RequestT[U, T, R] =
    header(HeaderNames.ContentType, mt.toString, replaceExisting = true)
  def contentType(ct: String, encoding: String): RequestT[U, T, R] =
    header(HeaderNames.ContentType, contentTypeWithCharset(ct, encoding), replaceExisting = true)
  def contentLength(l: Long): RequestT[U, T, R] =
    header(HeaderNames.ContentLength, l.toString, replaceExisting = true)

  /** Adds the given header to the end of the headers sequence.
    * @param replaceExisting
    *   If there's already a header with the same name, should it be replaced?
    */
  def header(h: Header, replaceExisting: Boolean = false): RequestT[U, T, R] = {
    val current = if (replaceExisting) headers.filterNot(_.is(h.name)) else headers
    this.copy(headers = current :+ h)
  }

  /** Adds the given header to the end of the headers sequence.
    * @param replaceExisting
    *   If there's already a header with the same name, should it be replaced?
    */
  def header(k: String, v: String, replaceExisting: Boolean): RequestT[U, T, R] =
    header(Header(k, v), replaceExisting)

  /** Adds the given header to the end of the headers sequence. */
  def header(k: String, v: String): RequestT[U, T, R] = header(Header(k, v))

  /** Adds the given header to the end of the headers sequence, if the value is defined. Otherwise has no effect. */
  def header(k: String, ov: Option[String]): RequestT[U, T, R] = ov.fold(this)(header(k, _))

  /** Adds the given headers to the end of the headers sequence. */
  def headers(hs: Map[String, String]): RequestT[U, T, R] = headers(hs.map(t => Header(t._1, t._2)).toSeq: _*)

  /** Adds the given headers to the end of the headers sequence.
    * @param replaceExisting
    *   If there's already a header with the same name, should it be replaced?
    */
  def headers(hs: Map[String, String], replaceExisting: Boolean): RequestT[U, T, R] =
    if (replaceExisting) hs.foldLeft(this) { (s, h) => s.header(h._1, h._2, replaceExisting) }
    else headers(hs)

  /** Adds the given headers to the end of the headers sequence. */
  def headers(hs: Header*): RequestT[U, T, R] = this.copy(headers = headers ++ hs)

  /** Adds the given headers to the end of the headers sequence. */
  def headers(hs: Seq[Header], replaceExisting: Boolean): RequestT[U, T, R] =
    if (replaceExisting) hs.foldLeft(this) { (s, h) => s.header(h, replaceExisting) }
    else headers(hs: _*)

  def auth: SpecifyAuthScheme[U, T, R] =
    new SpecifyAuthScheme[U, T, R](HeaderNames.Authorization, this, DigestAuthenticationBackend.DigestAuthTag)
  def proxyAuth: SpecifyAuthScheme[U, T, R] =
    new SpecifyAuthScheme[U, T, R](HeaderNames.ProxyAuthorization, this, DigestAuthenticationBackend.ProxyDigestAuthTag)
  def acceptEncoding(encoding: String): RequestT[U, T, R] =
    header(HeaderNames.AcceptEncoding, encoding, replaceExisting = true)

  def cookie(nv: (String, String)): RequestT[U, T, R] = cookies(nv)
  def cookie(n: String, v: String): RequestT[U, T, R] = cookies((n, v))
  def cookies(r: Response[_]): RequestT[U, T, R] = cookies(
    r.cookies.collect { case Right(c) => c }.map(c => (c.name, c.value)): _*
  )
  def cookies(cs: Iterable[CookieWithMeta]): RequestT[U, T, R] = cookies(cs.map(c => (c.name, c.value)).toSeq: _*)
  def cookies(nvs: (String, String)*): RequestT[U, T, R] = {
    header(
      HeaderNames.Cookie,
      (headers.find(_.is(HeaderNames.Cookie)).map(_.value).toSeq ++ nvs.map(p => p._1 + "=" + p._2)).mkString("; "),
      replaceExisting = true
    )
  }

  /** Uses the `utf-8` encoding.
    *
    * If content type is not yet specified, will be set to `text/plain` with `utf-8` encoding.
    *
    * If content length is not yet specified, will be set to the number of bytes in the string using the `utf-8`
    * encoding.
    */
  def body(b: String): RequestT[U, T, R] = body(b, Utf8)

  /** If content type is not yet specified, will be set to `text/plain` with the given encoding.
    *
    * If content length is not yet specified, will be set to the number of bytes in the string using the given encoding.
    */
  def body(b: String, encoding: String): RequestT[U, T, R] =
    withBody(StringBody(b, encoding))
      .setContentLengthIfMissing(b.getBytes(encoding).length.toLong)

  /** If content type is not yet specified, will be set to `application/octet-stream`.
    *
    * If content length is not yet specified, will be set to the length of the given array.
    */
  def body(b: Array[Byte]): RequestT[U, T, R] =
    withBody(ByteArrayBody(b))
      .setContentLengthIfMissing(b.length.toLong)

  /** If content type is not yet specified, will be set to `application/octet-stream`.
    */
  def body(b: ByteBuffer): RequestT[U, T, R] =
    withBody(ByteBufferBody(b))

  /** If content type is not yet specified, will be set to `application/octet-stream`.
    */
  def body(b: InputStream): RequestT[U, T, R] =
    withBody(InputStreamBody(b))

  /** If content type is not yet specified, will be set to `application/octet-stream`.
    *
    * If content length is not yet specified, will be set to the length of the given file.
    */
  private[client3] def body(f: SttpFile): RequestT[U, T, R] =
    withBody(FileBody(f)).setContentLengthIfMissing(f.size)

  /** Encodes the given parameters as form data using `utf-8`. If content type is not yet specified, will be set to
    * `application/x-www-form-urlencoded`.
    *
    * If content length is not yet specified, will be set to the length of the number of bytes in the url-encoded
    * parameter string.
    */
  def body(fs: Map[String, String]): RequestT[U, T, R] =
    formDataBody(fs.toList, Utf8)

  /** Encodes the given parameters as form data. If content type is not yet specified, will be set to
    * `application/x-www-form-urlencoded`.
    *
    * If content length is not yet specified, will be set to the length of the number of bytes in the url-encoded
    * parameter string.
    */
  def body(fs: Map[String, String], encoding: String): RequestT[U, T, R] =
    formDataBody(fs.toList, encoding)

  /** Encodes the given parameters as form data using `utf-8`. If content type is not yet specified, will be set to
    * `application/x-www-form-urlencoded`.
    *
    * If content length is not yet specified, will be set to the length of the number of bytes in the url-encoded
    * parameter string.
    */
  def body(fs: (String, String)*): RequestT[U, T, R] =
    formDataBody(fs.toList, Utf8)

  /** Encodes the given parameters as form data. If content type is not yet specified, will be set to
    * `application/x-www-form-urlencoded`.
    *
    * If content length is not yet specified, will be set to the length of the number of bytes in the url-encoded
    * parameter string.
    */
  def body(fs: Seq[(String, String)], encoding: String): RequestT[U, T, R] =
    formDataBody(fs, encoding)

  def multipartBody[R2](ps: Seq[Part[RequestBody[R2]]]): RequestT[U, T, R with R2] =
    copy(body = MultipartBody(ps))

  def multipartBody[R2](p1: Part[RequestBody[R2]], ps: Part[RequestBody[R2]]*): RequestT[U, T, R with R2] =
    copy(body = MultipartBody(p1 :: ps.toList))

  def streamBody[S](s: Streams[S])(b: s.BinaryStream): RequestT[U, T, R with S] =
    withBody(StreamBody(s)(b))

  /** When the request is sent, if reading the response times out (there's no activity for the given period of time), a
    * failed effect will be returned, or an exception will be thrown
    */
  def readTimeout(t: Duration): RequestT[U, T, R] =
    this.copy(options = options.copy(readTimeout = t))

  /** Specifies the target type to which the response body should be read. Note that this replaces any previous
    * specifications, which also includes any previous `mapResponse` invocations.
    */
  def response[T2, R2](ra: ResponseAs[T2, R2]): RequestT[U, T2, R with R2] =
    this.copy(response = ra)

  def mapResponse[T2](f: T => T2): RequestT[U, T2, R] =
    this.copy(response = response.map(f))

  def isWebSocket: Boolean = ResponseAs.isWebSocket(response)

  def followRedirects(fr: Boolean): RequestT[U, T, R] =
    this.copy(options = options.copy(followRedirects = fr))

  def maxRedirects(n: Int): RequestT[U, T, R] =
    if (n <= 0)
      this.copy(options = options.copy(followRedirects = false))
    else
      this.copy(options = options.copy(followRedirects = true, maxRedirects = n))

  def tag(k: String, v: Any): RequestT[U, T, R] =
    this.copy(tags = tags + (k -> v))

  def tag(k: String): Option[Any] = tags.get(k)

  private val disableAutoDecompressionKey = "disableAutoDecompression"

  // Used as a workaround to keep binary compatibility
  // TODO: replace with additional parameter in RequestOptions when writing sttp4

  def disableAutoDecompression: RequestT[U, T, R] = tag(disableAutoDecompressionKey, true)

  def autoDecompressionDisabled: Boolean = tags.getOrElse(disableAutoDecompressionKey, false).asInstanceOf[Boolean]

  private val httpVersionKey = "httpVersion"

  // Used as a workaround to keep binary compatibility
  // TODO: replace with additional parameter in RequestOptions when writing sttp4
  // TODO: add similar functionality to Response

  /** Allows setting HTTP version per request. Supported only is a few backends
    *
    * @param version:
    *   one of values from [[HttpVersion]] enum.
    * @return
    *   request with version tag
    */
  def httpVersion(version: HttpVersion): RequestT[U, T, R] = tag(httpVersionKey, version)

  /** Get[[HttpVersion]] from tags in request. Supported only is a few backends
    *
    * @return
    *   one of values form [[HttpVersion]] enum or [[None]]
    */
  def httpVersion: Option[HttpVersion] = tags.get(httpVersionKey).map(_.asInstanceOf[HttpVersion])

  private val loggingOptionsTagKey = "loggingOptions"

  /** Will only have effect when using the `LoggingBackend` */
  def logSettings(
      logRequestBody: Option[Boolean] = None,
      logResponseBody: Option[Boolean] = None,
      logRequestHeaders: Option[Boolean] = None,
      logResponseHeaders: Option[Boolean] = None
  ): RequestT[U, T, R] = {
    val loggingOptions = LoggingOptions(
      logRequestBody = logRequestBody,
      logResponseBody = logResponseBody,
      logRequestHeaders = logRequestHeaders,
      logResponseHeaders = logResponseHeaders
    )
    this.tag(loggingOptionsTagKey, loggingOptions)
  }

  def loggingOptions: Option[LoggingOptions] = tag(loggingOptionsTagKey).asInstanceOf[Option[LoggingOptions]]

  /** When a POST or PUT request is redirected, should the redirect be a POST/PUT as well (with the original body), or
    * should the request be converted to a GET without a body.
    *
    * Note that this only affects 301 and 302 redirects. 303 redirects are always converted, while 307 and 308 redirects
    * always keep the same method.
    *
    * See https://developer.mozilla.org/en-US/docs/Web/HTTP/Redirections for details.
    */
  def redirectToGet(r: Boolean): RequestT[U, T, R] =
    this.copy(options = options.copy(redirectToGet = r))

  /** Sends the request, using the backend from the implicit scope. Only requests for which the method & URI are
    * specified can be sent.
    *
    * The required capabilities must be a subset of the capabilities provided by the backend.
    *
    * @return
    *   For synchronous backends (when the effect type is [[Identity]]), [[Response]] is returned directly and
    *   exceptions are thrown. For asynchronous backends (when the effect type is e.g. [[scala.concurrent.Future]]), an
    *   effect containing the [[Response]] is returned. Exceptions are represented as failed effects (e.g. failed
    *   futures).
    *
    * The response body is deserialized as specified by this request (see [[RequestT.response]]).
    *
    * Known exceptions are converted by backends to one of [[SttpClientException]]. Other exceptions are thrown
    * unchanged.
    */
  @deprecated(message = "use request.send(backend), providing the backend explicitly", since = "3.0.0")
  def send[F[_], P]()(implicit
      backend: SttpBackend[F, P],
      pEffectFIsR: P with Effect[F] <:< R,
      isIdInRequest: IsIdInRequest[U]
  ): F[Response[T]] =
    send(backend)(
      isIdInRequest,
      pEffectFIsR
    ) // the order of implicits must be different so that the signatures are different

  /** Sends the request, using the given backend. Only requests for which the method & URI are specified can be sent.
    *
    * The required capabilities must be a subset of the capabilities provided by the backend.
    *
    * @return
    *   For synchronous backends (when the effect type is [[Identity]]), [[Response]] is returned directly and
    *   exceptions are thrown. For asynchronous backends (when the effect type is e.g. [[scala.concurrent.Future]]), an
    *   effect containing the [[Response]] is returned. Exceptions are represented as failed effects (e.g. failed
    *   futures).
    *
    * The response body is deserialized as specified by this request (see [[RequestT.response]]).
    *
    * Known exceptions are converted by backends to one of [[SttpClientException]]. Other exceptions are thrown
    * unchanged.
    */
  def send[F[_], P](backend: SttpBackend[F, P])(implicit
      isIdInRequest: IsIdInRequest[U],
      pEffectFIsR: P with Effect[F] <:< R
  ): F[Response[T]] = backend.send(asRequest.asInstanceOf[Request[T, P with Effect[F]]]) // as witnessed by pEffectFIsR

  def toCurl(implicit isIdInRequest: IsIdInRequest[U]): String = ToCurlConverter.requestToCurl(asRequest)
  def toCurl(sensitiveHeaders: Set[String])(implicit isIdInRequest: IsIdInRequest[U]): String =
    ToCurlConverter.requestToCurl(asRequest, sensitiveHeaders)

  def toRfc2616Format(implicit isIdInRequest: IsIdInRequest[U]): String = ToRfc2616Converter.requestToRfc2616(asRequest)
  def toRfc2616Format(sensitiveHeaders: Set[String])(implicit isIdInRequest: IsIdInRequest[U]): String =
    ToRfc2616Converter.requestToRfc2616(asRequest, sensitiveHeaders)

  def showBasic: String =
    (this.method, this.uri) match {
      case (m: Method, u: Uri) =>
        val ws = if (isWebSocket) " (web socket) " else ""
        s"$m$ws $u"
      case _ => "(no method & uri set)"
    }

  def show(
      includeBody: Boolean = true,
      includeHeaders: Boolean = true,
      sensitiveHeaders: Set[String] = HeaderNames.SensitiveHeaders
  ): String = {
    val headers =
      if (includeHeaders) ", headers: " + this.headers.map(_.toStringSafe(sensitiveHeaders)).mkString(", ") else ""
    val body = if (includeBody) s", body: ${this.body.show}" else ""
    val methodAndUri = (this.method, this.uri) match {
      case (m: Method, u: Uri) =>
        val ws = if (isWebSocket) " (web socket) " else ""
        s"$m$ws $u, "
      case _ => ""
    }
    s"${methodAndUri}response as: ${response.show}$headers$body"
  }

  private def asRequest(implicit isIdInRequest: IsIdInRequest[U]): RequestT[Identity, T, R] = {
    // we could avoid the asInstanceOf by creating an artificial copy
    // changing the method & url fields using `isIdInRequest`, but that
    // would be only to satisfy the type checker, and a needless copy at
    // runtime.
    this.asInstanceOf[RequestT[Identity, T, R]]
  }

  private def hasContentType: Boolean = headers.exists(_.is(HeaderNames.ContentType))
  private def setContentTypeIfMissing(mt: MediaType): RequestT[U, T, R] =
    if (hasContentType) this else contentType(mt)

  private[client3] def withBody[R2](body: RequestBody[R2]): RequestT[U, T, R with R2] = {
    val defaultCt = body match {
      case StringBody(_, encoding, ct) =>
        ct.copy(charset = Some(encoding))
      case _ =>
        body.defaultContentType
    }

    setContentTypeIfMissing(defaultCt).copy(body = body)
  }

  private def hasContentLength: Boolean = headers.exists(_.is(HeaderNames.ContentLength))
  private def setContentLengthIfMissing(l: => Long): RequestT[U, T, R] =
    if (hasContentLength) this else contentLength(l)

  private def formDataBody(fs: Seq[(String, String)], encoding: String): RequestT[U, T, R] = {
    val b = RequestBody.paramsToStringBody(fs, encoding)
    setContentTypeIfMissing(MediaType.ApplicationXWwwFormUrlencoded)
      .setContentLengthIfMissing(b.s.getBytes(encoding).length.toLong)
      .copy(body = b)
  }

  private[client3] def onlyMetadata(implicit isIdInRequest: IsIdInRequest[U]): RequestMetadata = {
    val r = asRequest
    val m = r.method
    val u = r.uri
    val h = r.headers
    new RequestMetadata {
      override val method: Method = m
      override val uri: Uri = u
      override val headers: Seq[Header] = h
    }
  }
}

object RequestT {
  implicit class RichRequestTEither[U[_], A, B, R](r: RequestT[U, Either[A, B], R]) {
    def mapResponseRight[B2](f: B => B2): RequestT[U, Either[A, B2], R] = r.copy(response = r.response.mapRight(f))
    def responseGetRight: RequestT[U, B, R] = r.copy(response = r.response.getRight)
  }

  implicit class RichRequestTEitherResponseException[U[_], HE, DE, B, R](
      r: RequestT[U, Either[ResponseException[HE, DE], B], R]
  ) {
    def responseGetEither: RequestT[U, Either[HE, B], R] = r.copy(response = r.response.getEither)
  }
}

class SpecifyAuthScheme[U[_], T, -R](hn: String, rt: RequestT[U, T, R], digestTag: String) {
  def basic(user: String, password: String): RequestT[U, T, R] = {
    val c = new String(Base64.getEncoder.encode(s"$user:$password".getBytes(Utf8)), Utf8)
    rt.header(hn, s"Basic $c")
  }

  def basicToken(token: String): RequestT[U, T, R] =
    rt.header(hn, s"Basic $token")

  def bearer(token: String): RequestT[U, T, R] =
    rt.header(hn, s"Bearer $token")

  def digest(user: String, password: String): RequestT[U, T, R] = {
    rt.tag(digestTag, DigestAuthData(user, password))
  }
}

case class RequestOptions(
    followRedirects: Boolean,
    readTimeout: Duration, // TODO: Use FiniteDuration while migrating to sttp-4
    maxRedirects: Int,
    redirectToGet: Boolean
)




© 2015 - 2025 Weber Informatics LLC | Privacy Policy