sttp.client4.asynchttpclient.fs2.AsyncHttpClientFs2Backend.scala Maven / Gradle / Ivy
package sttp.client4.asynchttpclient.fs2
import cats.effect._
import cats.effect.implicits._
import cats.implicits._
import fs2.concurrent.InspectableQueue
import fs2.interop.reactivestreams._
import fs2.{Chunk, Pipe, Stream}
import io.netty.buffer.{ByteBuf, Unpooled}
import org.asynchttpclient.{Request => _, Response => _, _}
import org.reactivestreams.Publisher
import sttp.capabilities.WebSockets
import sttp.capabilities.fs2.Fs2Streams
import sttp.client4.asynchttpclient.{AsyncHttpClientBackend, BodyFromAHC, BodyToAHC}
import sttp.client4.impl.cats.CatsMonadAsyncError
import sttp.client4.impl.fs2.{Fs2SimpleQueue, Fs2WebSockets}
import sttp.client4.internal._
import sttp.client4.internal.ws.SimpleQueue
import sttp.client4.testing.WebSocketStreamBackendStub
import sttp.client4._
import sttp.client4.wrappers.FollowRedirectsBackend
import sttp.monad.MonadAsyncError
import sttp.ws.{WebSocket, WebSocketFrame}
import java.io.File
import java.nio.ByteBuffer
import scala.collection.immutable.Queue
class AsyncHttpClientFs2Backend[F[_]: ConcurrentEffect: ContextShift] private (
asyncHttpClient: AsyncHttpClient,
closeClient: Boolean,
blocker: Blocker,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder,
webSocketBufferCapacity: Option[Int]
) extends AsyncHttpClientBackend[F, Fs2Streams[F], Fs2Streams[F] with WebSockets](
asyncHttpClient,
new CatsMonadAsyncError,
closeClient,
customizeRequest
)
with WebSocketStreamBackend[F, Fs2Streams[F]] {
override val streams: Fs2Streams[F] = Fs2Streams[F]
override def send[T](r: GenericRequest[T, R]): F[Response[T]] =
super.send(r).guarantee(implicitly[ContextShift[F]].shift)
override protected val bodyFromAHC: BodyFromAHC[F, Fs2Streams[F]] =
new BodyFromAHC[F, Fs2Streams[F]] {
override val streams: Fs2Streams[F] = Fs2Streams[F]
override implicit val monad: MonadAsyncError[F] = new CatsMonadAsyncError
override def publisherToStream(p: Publisher[ByteBuffer]): Stream[F, Byte] =
p.toStream[F].flatMap(buf => Stream.chunk(Chunk.byteBuffer(buf)))
override def publisherToBytes(p: Publisher[ByteBuffer]): F[Array[Byte]] =
p.toStream[F]
.compile
.fold(Queue.empty[Array[Byte]])(enqueueBytes)
.map(concatBytes)
override def publisherToFile(p: Publisher[ByteBuffer], f: File): F[Unit] =
p.toStream[F]
.flatMap(b => Stream.emits(b.array()))
.through(fs2.io.file.writeAll(f.toPath, blocker))
.compile
.drain
override def bytesToPublisher(b: Array[Byte]): F[Publisher[ByteBuffer]] =
(Stream.apply[F, ByteBuffer](ByteBuffer.wrap(b)).toUnicastPublisher: Publisher[ByteBuffer]).pure[F]
override def fileToPublisher(f: File): F[Publisher[ByteBuffer]] =
(fs2.io.file
.readAll(f.toPath, blocker, IOBufferSize)
.mapChunks(c => Chunk(ByteBuffer.wrap(c.toArray)))
.toUnicastPublisher: Publisher[ByteBuffer]).pure[F]
override def compileWebSocketPipe(
ws: WebSocket[F],
pipe: Pipe[F, WebSocketFrame.Data[_], WebSocketFrame]
): F[Unit] = Fs2WebSockets.handleThroughPipe(ws)(pipe)
}
override protected val bodyToAHC: BodyToAHC[F, Fs2Streams[F]] = new BodyToAHC[F, Fs2Streams[F]] {
override val streams: Fs2Streams[F] = Fs2Streams[F]
override protected def streamToPublisher(s: Stream[F, Byte]): Publisher[ByteBuf] =
s.chunks.map(c => Unpooled.wrappedBuffer(c.toArray)).toUnicastPublisher
}
override protected def createSimpleQueue[T]: F[SimpleQueue[F, T]] =
webSocketBufferCapacity
.fold(InspectableQueue.unbounded[F, T])(InspectableQueue.bounded)
.map(new Fs2SimpleQueue(_, webSocketBufferCapacity))
}
object AsyncHttpClientFs2Backend {
private def apply[F[_]: ConcurrentEffect: ContextShift](
asyncHttpClient: AsyncHttpClient,
closeClient: Boolean,
blocker: Blocker,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder,
webSocketBufferCapacity: Option[Int]
): WebSocketStreamBackend[F, Fs2Streams[F]] =
FollowRedirectsBackend(
new AsyncHttpClientFs2Backend(asyncHttpClient, closeClient, blocker, customizeRequest, webSocketBufferCapacity)
)
def apply[F[_]: ConcurrentEffect: ContextShift](
blocker: Blocker,
options: BackendOptions = BackendOptions.Default,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): F[WebSocketStreamBackend[F, Fs2Streams[F]]] =
implicitly[Sync[F]]
.delay(
apply[F](
AsyncHttpClientBackend.defaultClient(options),
closeClient = true,
blocker,
customizeRequest,
webSocketBufferCapacity
)
)
/** Makes sure the backend is closed after usage.
*/
def resource[F[_]: ConcurrentEffect: ContextShift](
blocker: Blocker,
options: BackendOptions = BackendOptions.Default,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): Resource[F, WebSocketStreamBackend[F, Fs2Streams[F]]] =
Resource.make(apply(blocker, options, customizeRequest, webSocketBufferCapacity))(_.close())
def usingConfig[F[_]: ConcurrentEffect: ContextShift](
blocker: Blocker,
cfg: AsyncHttpClientConfig,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): F[WebSocketStreamBackend[F, Fs2Streams[F]]] =
implicitly[Sync[F]].delay(
apply[F](new DefaultAsyncHttpClient(cfg), closeClient = true, blocker, customizeRequest, webSocketBufferCapacity)
)
/** Makes sure the backend is closed after usage.
*/
def resourceUsingConfig[F[_]: ConcurrentEffect: ContextShift](
cfg: AsyncHttpClientConfig,
blocker: Blocker,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): Resource[F, WebSocketStreamBackend[F, Fs2Streams[F]]] =
Resource.make(usingConfig(blocker, cfg, customizeRequest, webSocketBufferCapacity))(_.close())
/** @param updateConfig
* A function which updates the default configuration (created basing on `options`).
*/
def usingConfigBuilder[F[_]: ConcurrentEffect: ContextShift](
blocker: Blocker,
updateConfig: DefaultAsyncHttpClientConfig.Builder => DefaultAsyncHttpClientConfig.Builder,
options: BackendOptions = BackendOptions.Default,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): F[WebSocketStreamBackend[F, Fs2Streams[F]]] =
implicitly[Sync[F]].delay(
AsyncHttpClientFs2Backend[F](
AsyncHttpClientBackend.clientWithModifiedOptions(options, updateConfig),
closeClient = true,
blocker,
customizeRequest,
webSocketBufferCapacity
)
)
/** Makes sure the backend is closed after usage.
* @param updateConfig
* A function which updates the default configuration (created basing on `options`).
*/
def resourceUsingConfigBuilder[F[_]: ConcurrentEffect: ContextShift](
blocker: Blocker,
updateConfig: DefaultAsyncHttpClientConfig.Builder => DefaultAsyncHttpClientConfig.Builder,
options: BackendOptions = BackendOptions.Default,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): Resource[F, WebSocketStreamBackend[F, Fs2Streams[F]]] =
Resource.make(usingConfigBuilder(blocker, updateConfig, options, customizeRequest, webSocketBufferCapacity))(
_.close()
)
def usingClient[F[_]: ConcurrentEffect: ContextShift](
client: AsyncHttpClient,
blocker: Blocker,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): WebSocketStreamBackend[F, Fs2Streams[F]] =
apply[F](client, closeClient = false, blocker, customizeRequest, webSocketBufferCapacity)
/** Create a stub backend for testing, which uses the `F` response wrapper, and supports `Stream[F, ByteBuffer]`
* streaming.
*
* See [[WebSocketStreamBackendStub]] for details on how to configure stub responses.
*/
def stub[F[_]: Concurrent]: WebSocketStreamBackendStub[F, Fs2Streams[F]] =
WebSocketStreamBackendStub(new CatsMonadAsyncError())
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy