org.http4s.client.Client.scala Maven / Gradle / Ivy
/*
* Copyright 2014 http4s.org
*
* 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
*
* http://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 org.http4s
package client
import cats.data._
import cats.effect._
import cats.effect.implicits.genSpawnOps
import cats.effect.implicits.monadCancelOps_
import cats.effect.kernel.CancelScope
import cats.effect.kernel.Poll
import cats.effect.kernel.Resource
import cats.syntax.all._
import cats.~>
import fs2._
import fs2.concurrent.Channel
import org.http4s.headers.Host
import java.io.IOException
import scala.util.control.NoStackTrace
/** A [[Client]] submits [[Request]]s to a server and processes the [[Response]].
*
* When a connection is "released" and the HTTP semantics of the
* request and response permit, the connection may be kept alive by
* the backend and used for a subsequent request. When HTTP
* semantics require it, or at the backend's discretion, a released
* connection may also be closed.
*/
trait Client[F[_]] {
def run(req: Request[F]): Resource[F, Response[F]]
/** Returns this client as a [[cats.data.Kleisli]]. All connections created
* by this service are released on completion of callback task f.
*
* This method effectively reverses the arguments to [[run]] followed by `use`, and is
* preferred when an HTTP client is composed into a larger Kleisli function,
* or when a common response callback is used by many call sites.
*/
def toKleisli[A](f: Response[F] => F[A]): Kleisli[F, Request[F], A]
/** Returns this client as an [[HttpApp]]. It is the responsibility
* of callers of this service to run the response body to release
* the underlying HTTP connection.
*
* This is intended for use in proxy servers. [[run]], [[fetchAs[A](req:org\.http4s\.Request[F])*]],
* [[toKleisli]] are safer alternatives, as their
* signatures guarantee release of the HTTP connection.
*/
def toHttpApp: HttpApp[F]
/** Run the request as a stream. The response lifecycle is equivalent
* to the returned Stream's.
*/
def stream(req: Request[F]): Stream[F, Response[F]]
def expectOr[A](req: Request[F])(onError: Response[F] => F[Throwable])(implicit
d: EntityDecoder[F, A]
): F[A]
/** Submits a request and decodes the response on success. On failure, the
* status code is returned. The underlying HTTP connection is released at the
* completion of the decoding.
*/
def expect[A](req: Request[F])(implicit d: EntityDecoder[F, A]): F[A]
def expectOr[A](req: F[Request[F]])(onError: Response[F] => F[Throwable])(implicit
d: EntityDecoder[F, A]
): F[A]
@deprecated("Use req.flatMap(expect(_))", "0.23.16")
def expect[A](req: F[Request[F]])(implicit d: EntityDecoder[F, A]): F[A]
def expectOr[A](uri: Uri)(onError: Response[F] => F[Throwable])(implicit
d: EntityDecoder[F, A]
): F[A]
/** Submits a GET request to the specified URI and decodes the response on
* success. On failure, the status code is returned. The underlying HTTP
* connection is released at the completion of the decoding.
*/
def expect[A](uri: Uri)(implicit d: EntityDecoder[F, A]): F[A]
def expectOr[A](s: String)(onError: Response[F] => F[Throwable])(implicit
d: EntityDecoder[F, A]
): F[A]
/** Submits a GET request to the URI specified by the String and decodes the
* response on success. On failure, the status code is returned. The
* underlying HTTP connection is released at the completion of the decoding.
*/
def expect[A](s: String)(implicit d: EntityDecoder[F, A]): F[A]
def expectOptionOr[A](req: Request[F])(onError: Response[F] => F[Throwable])(implicit
d: EntityDecoder[F, A]
): F[Option[A]]
def expectOption[A](req: Request[F])(implicit d: EntityDecoder[F, A]): F[Option[A]]
/** Submits a request and decodes the response, regardless of the status code.
* The underlying HTTP connection is released at the completion of the
* decoding.
*/
def fetchAs[A](req: Request[F])(implicit d: EntityDecoder[F, A]): F[A]
/** Submits a request and decodes the response, regardless of the status code.
* The underlying HTTP connection is released at the completion of the
* decoding.
*/
@deprecated("Use req.flatMap(fetchAs(_))", "0.23.16")
def fetchAs[A](req: F[Request[F]])(implicit d: EntityDecoder[F, A]): F[A]
/** Submits a request and returns the response status */
def status(req: Request[F]): F[Status]
/** Submits a request and returns the response status */
@deprecated("Use req.flatMap(status(_))", "0.23.16")
def status(req: F[Request[F]]): F[Status]
/** Submits a GET request to the URI and returns the response status */
def statusFromUri(uri: Uri): F[Status]
/** Submits a GET request to the URI and returns the response status */
def statusFromString(s: String): F[Status]
/** Submits a request and returns true if and only if the response status is
* successful
*/
def successful(req: Request[F]): F[Boolean]
/** Submits a request and returns true if and only if the response status is
* successful
*/
@deprecated("Use req.flatMap(successful(_))", "0.23.16")
def successful(req: F[Request[F]]): F[Boolean]
/** Submits a GET request, and provides a callback to process the response.
*
* @param uri The URI to GET
* @param f A callback for the response to a GET on uri. The underlying HTTP connection
* is released when the returned task completes. Attempts to read the
* response body afterward will result in an error.
* @return The result of applying f to the response to req
*/
def get[A](uri: Uri)(f: Response[F] => F[A]): F[A]
/** Submits a request and decodes the response on success. On failure, the
* status code is returned. The underlying HTTP connection is released at the
* completion of the decoding.
*/
def get[A](s: String)(f: Response[F] => F[A]): F[A]
/** As [[expectOptionOr]], but defined in terms of [[cats.data.OptionT]]. */
final def expectOptionOrT[A](req: Request[F])(onError: Response[F] => F[Throwable])(implicit
d: EntityDecoder[F, A]
): OptionT[F, A] =
OptionT(expectOptionOr(req)(onError)(d))
/** As [[expectOption]], but defined in terms of [[cats.data.OptionT]]. */
final def expectOptionT[A](req: Request[F])(implicit d: EntityDecoder[F, A]): OptionT[F, A] =
OptionT(expectOption[A](req)(d))
/** Translates the effect type of this client from F to G
*/
def translate[G[_]](
fk: F ~> G
)(gK: G ~> F)(implicit F: MonadCancelThrow[F]): Client[G] = {
implicit val G: MonadCancelThrow[G] = liftMonadCancel(F)(fk)(gK)
Client[G]((req: Request[G]) =>
run(
req.mapK(gK)
).mapK(fk)
.map(_.mapK(fk))
)
}
private[this] def liftMonadCancel[G[_]](
F: MonadCancelThrow[F]
)(fk: F ~> G)(gk: G ~> F): MonadCancelThrow[G] =
new MonadCancelThrow[G] {
def pure[A](x: A): G[A] = fk(F.pure(x))
// Members declared in cats.ApplicativeError
def handleErrorWith[A](ga: G[A])(f: Throwable => G[A]): G[A] =
fk(F.handleErrorWith(gk(ga))(ex => gk(f(ex))))
def raiseError[A](e: Throwable): G[A] = fk(F.raiseError[A](e))
// Members declared in cats.FlatMap
def flatMap[A, B](ga: G[A])(f: A => G[B]): G[B] =
fk(F.flatMap(gk(ga))(a => gk(f(a))))
def tailRecM[A, B](a: A)(f: A => G[Either[A, B]]): G[B] =
fk(F.tailRecM(a)(a => gk(f(a))))
// Members declared in cats.effect.kernel.MonadCancel
def canceled: G[Unit] = fk(F.canceled)
def forceR[A, B](ga: G[A])(gb: G[B]): G[B] =
fk(F.forceR(gk(ga))(gk(gb)))
def onCancel[A](ga: G[A], fin: G[Unit]): G[A] =
fk(F.onCancel(gk(ga), gk(fin)))
def rootCancelScope: CancelScope = F.rootCancelScope
def uncancelable[A](body: Poll[G] => G[A]): G[A] =
fk(F.uncancelable { pollF =>
gk(body(new Poll[G] {
def apply[B](gb: G[B]): G[B] = fk(pollF(gk(gb)))
}))
})
}
}
object Client {
def apply[F[_]](
f: Request[F] => Resource[F, Response[F]]
)(implicit F: MonadCancelThrow[F]): Client[F] =
new DefaultClient[F] {
def run(req: Request[F]): Resource[F, Response[F]] = f(req)
}
/** Creates a client from the specified [[HttpApp]]. Useful for
* generating pre-determined responses for requests in testing.
*
* @param app the [[HttpApp]] to respond to requests to this client
*/
def fromHttpApp[F[_]](
app: HttpApp[F]
)(implicit F: Concurrent[F]): Client[F] = {
def until[A](disposed: Ref[F, Boolean])(source: Stream[F, A]): Stream[F, A] = {
def go(stream: Stream[F, A]): Pull[F, A, Unit] =
stream.pull.uncons.flatMap {
case Some((chunk, stream)) =>
Pull.eval(disposed.get).flatMap {
case true => Pull.raiseError[F](new IOException("response was disposed"))
case false => Pull.output(chunk) >> go(stream)
}
case None => Pull.done
}
go(source).stream
}
def processResponse(
response: Response[F],
disposed: Ref[F, Boolean],
): Resource[F, Response[F]] =
response.entity match {
case Entity.Empty | Entity.Strict(_) =>
Resource.pure(response)
case Entity.Streamed(_, _) =>
for {
channel <- Resource.eval(Channel.synchronous[F, Chunk[Byte]])
producer = response.body.chunks
.through(channel.sendAll)
.compile
.drain
.uncancelable
_ <- Resource.make(producer.start)(p =>
channel.stream.compile.drain.guarantee(p.join.void)
)
r = response.withBodyStream(
Stream
.eval(disposed.get)
.ifM(
Stream.raiseError[F](new IOException("response was disposed")),
channel.stream.unchunks
.onFinalize(
channel.stream.compile.drain
.guarantee(disposed.set(true))
),
)
)
_ <- Resource.onFinalize {
disposed.get
.ifM(
F.unit,
r.body.compile.drain,
)
}
} yield r
}
def run(req: Request[F]): Resource[F, Response[F]] =
Resource.eval(Ref[F].of(false)).flatMap { disposed =>
val reqAugmented =
addHostHeaderIfUriIsAbsolute(req.pipeBodyThrough(until(disposed)))
Resource
.eval(app(reqAugmented))
.onFinalize(disposed.set(true))
.flatMap(processResponse(_, disposed))
}
Client(run)
}
/** This method introduces an important way for the effectful backends to allow tracing. As Kleisli types
* form the backend of tracing and these transformations are non-trivial.
*/
def liftKleisli[F[_]: MonadCancelThrow, A](client: Client[F]): Client[Kleisli[F, A, *]] =
Client { (req: Request[Kleisli[F, A, *]]) =>
Resource.eval(Kleisli.ask[F, A]).flatMap { a =>
client
.run(req.mapK(Kleisli.applyK[F, A](a)))
.mapK(Kleisli.liftK[F, A])
.map(_.mapK(Kleisli.liftK[F, A]))
}
}
private def addHostHeaderIfUriIsAbsolute[F[_]](req: Request[F]): Request[F] =
req.uri.host match {
case Some(host) if !req.headers.contains[Host] =>
req.withHeaders(req.headers.put(Host(host.value, req.uri.port)))
case _ => req
}
}
final case class UnexpectedStatus(status: Status, requestMethod: Method, requestUri: Uri)
extends RuntimeException
with NoStackTrace {
override def getMessage: String =
s"unexpected HTTP status: $status for request $requestMethod $requestUri"
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy