sttp.client4.asynchttpclient.zio.AsyncHttpClientZioBackend.scala Maven / Gradle / Ivy
package sttp.client4.asynchttpclient.zio
import _root_.zio._
import _root_.zio.blocking.Blocking
import _root_.zio.interop.reactivestreams.{
publisherToStream => publisherToZioStream,
streamToPublisher => zioStreamToPublisher
}
import _root_.zio.stream._
import io.netty.buffer.{ByteBuf, Unpooled}
import org.asynchttpclient._
import org.reactivestreams.Publisher
import sttp.capabilities.WebSockets
import sttp.capabilities.zio.ZioStreams
import sttp.client4.asynchttpclient.{AsyncHttpClientBackend, BodyFromAHC, BodyToAHC}
import sttp.client4.impl.zio.{RIOMonadAsyncError, ZioSimpleQueue, ZioWebSockets}
import sttp.client4.internal._
import sttp.client4.internal.ws.SimpleQueue
import sttp.client4.testing.WebSocketStreamBackendStub
import sttp.client4.wrappers.FollowRedirectsBackend
import sttp.client4.{wrappers, BackendOptions, WebSocketStreamBackend}
import sttp.monad.MonadAsyncError
import sttp.ws.{WebSocket, WebSocketFrame}
import java.io.{BufferedInputStream, File, FileInputStream, FileOutputStream}
import java.nio.ByteBuffer
import scala.collection.immutable
class AsyncHttpClientZioBackend private (
runtime: Runtime[Any],
asyncHttpClient: AsyncHttpClient,
closeClient: Boolean,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder,
webSocketBufferCapacity: Option[Int]
) extends AsyncHttpClientBackend[Task, ZioStreams, ZioStreams with WebSockets](
asyncHttpClient,
new RIOMonadAsyncError[Any],
closeClient,
customizeRequest
)
with WebSocketStreamBackend[Task, ZioStreams] {
override val streams: ZioStreams = ZioStreams
private val bufferSize = 16
override protected val bodyFromAHC: BodyFromAHC[Task, ZioStreams] =
new BodyFromAHC[Task, ZioStreams] {
override val streams: ZioStreams = ZioStreams
override implicit val monad: MonadAsyncError[Task] = new RIOMonadAsyncError[Any]
override def publisherToStream(p: Publisher[ByteBuffer]): Stream[Throwable, Byte] =
p.toStream(bufferSize).mapConcatChunk(Chunk.fromByteBuffer(_))
override def publisherToBytes(p: Publisher[ByteBuffer]): Task[Array[Byte]] =
p.toStream(bufferSize)
.fold(immutable.Queue.empty[Array[Byte]])(enqueueBytes)
.map(concatBytes)
override def publisherToFile(p: Publisher[ByteBuffer], f: File): Task[Unit] =
p.toStream(bufferSize)
.map(Chunk.fromByteBuffer)
.flattenChunks
.run(ZSink.fromOutputStream(new FileOutputStream(f)))
.unit
.provideLayer(Blocking.live)
override def bytesToPublisher(b: Array[Byte]): Task[Publisher[ByteBuffer]] =
Stream.apply(ByteBuffer.wrap(b)).toPublisher
override def fileToPublisher(f: File): Task[Publisher[ByteBuffer]] =
ZStream
.fromInputStream(new BufferedInputStream(new FileInputStream(f)))
.mapChunks(ch => Chunk(ByteBuffer.wrap(ch.toArray)))
.toPublisher
.provideLayer(Blocking.live)
override def compileWebSocketPipe(
ws: WebSocket[Task],
pipe: Stream[Throwable, WebSocketFrame.Data[_]] => Stream[Throwable, WebSocketFrame]
): Task[Unit] = ZioWebSockets.compilePipe(ws, pipe)
}
override protected val bodyToAHC: BodyToAHC[Task, ZioStreams] = new BodyToAHC[Task, ZioStreams] {
override val streams: ZioStreams = ZioStreams
override protected def streamToPublisher(s: Stream[Throwable, Byte]): Publisher[ByteBuf] =
runtime.unsafeRun(s.mapChunks(c => Chunk.single(Unpooled.wrappedBuffer(c.toArray))).toPublisher)
}
override protected def createSimpleQueue[T]: Task[SimpleQueue[Task, T]] =
for {
runtime <- ZIO.runtime[Any]
queue <- webSocketBufferCapacity match {
case Some(capacity) => Queue.dropping[T](capacity)
case None => Queue.unbounded[T]
}
} yield new ZioSimpleQueue(queue, runtime)
}
object AsyncHttpClientZioBackend {
private def apply[R](
runtime: Runtime[R],
asyncHttpClient: AsyncHttpClient,
closeClient: Boolean,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder,
webSocketBufferCapacity: Option[Int]
): WebSocketStreamBackend[Task, ZioStreams] =
wrappers.FollowRedirectsBackend(
new AsyncHttpClientZioBackend(runtime, asyncHttpClient, closeClient, customizeRequest, webSocketBufferCapacity)
)
def apply(
options: BackendOptions = BackendOptions.Default,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): Task[WebSocketStreamBackend[Task, ZioStreams]] =
ZIO
.runtime[Any]
.flatMap(runtime =>
Task.effect(
AsyncHttpClientZioBackend(
runtime,
AsyncHttpClientBackend.defaultClient(options),
closeClient = true,
customizeRequest,
webSocketBufferCapacity
)
)
)
def managed(
options: BackendOptions = BackendOptions.Default,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): TaskManaged[WebSocketStreamBackend[Task, ZioStreams]] =
ZManaged.make(apply(options, customizeRequest, webSocketBufferCapacity))(_.close().ignore)
def layer(
options: BackendOptions = BackendOptions.Default,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): Layer[Throwable, SttpClient] =
ZLayer.fromManaged(managed(options, customizeRequest, webSocketBufferCapacity))
def usingConfig(
cfg: AsyncHttpClientConfig,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): Task[WebSocketStreamBackend[Task, ZioStreams]] =
ZIO
.runtime[Any]
.flatMap(runtime =>
Task.effect(
AsyncHttpClientZioBackend(
runtime,
new DefaultAsyncHttpClient(cfg),
closeClient = true,
customizeRequest,
webSocketBufferCapacity
)
)
)
def managedUsingConfig(
cfg: AsyncHttpClientConfig,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): TaskManaged[WebSocketStreamBackend[Task, ZioStreams]] =
ZManaged.make(usingConfig(cfg, customizeRequest, webSocketBufferCapacity))(_.close().ignore)
def layerUsingConfig(
cfg: AsyncHttpClientConfig,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): Layer[Throwable, SttpClient] =
ZLayer.fromManaged(managedUsingConfig(cfg, customizeRequest, webSocketBufferCapacity))
/** @param updateConfig
* A function which updates the default configuration (created basing on `options`).
*/
def usingConfigBuilder(
updateConfig: DefaultAsyncHttpClientConfig.Builder => DefaultAsyncHttpClientConfig.Builder,
options: BackendOptions = BackendOptions.Default,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): Task[WebSocketStreamBackend[Task, ZioStreams]] =
ZIO
.runtime[Any]
.flatMap(runtime =>
Task.effect(
AsyncHttpClientZioBackend(
runtime,
AsyncHttpClientBackend.clientWithModifiedOptions(options, updateConfig),
closeClient = true,
customizeRequest,
webSocketBufferCapacity
)
)
)
/** @param updateConfig
* A function which updates the default configuration (created basing on `options`).
*/
def managedUsingConfigBuilder(
updateConfig: DefaultAsyncHttpClientConfig.Builder => DefaultAsyncHttpClientConfig.Builder,
options: BackendOptions = BackendOptions.Default,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): TaskManaged[WebSocketStreamBackend[Task, ZioStreams]] =
ZManaged.make(usingConfigBuilder(updateConfig, options, customizeRequest, webSocketBufferCapacity))(
_.close().ignore
)
/** @param updateConfig
* A function which updates the default configuration (created basing on `options`).
*/
def layerUsingConfigBuilder(
updateConfig: DefaultAsyncHttpClientConfig.Builder => DefaultAsyncHttpClientConfig.Builder,
options: BackendOptions = BackendOptions.Default,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): Layer[Throwable, SttpClient] =
ZLayer.fromManaged(managedUsingConfigBuilder(updateConfig, options, customizeRequest, webSocketBufferCapacity))
def usingClient[R](
runtime: Runtime[R],
client: AsyncHttpClient,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): WebSocketStreamBackend[Task, ZioStreams] =
AsyncHttpClientZioBackend(runtime, client, closeClient = false, customizeRequest, webSocketBufferCapacity)
def layerUsingClient(
client: AsyncHttpClient,
customizeRequest: BoundRequestBuilder => BoundRequestBuilder = identity,
webSocketBufferCapacity: Option[Int] = AsyncHttpClientBackend.DefaultWebSocketBufferCapacity
): Layer[Nothing, SttpClient] =
ZLayer.fromAcquireRelease(
UIO.runtime.map(runtime => usingClient(runtime, client, customizeRequest, webSocketBufferCapacity))
)(
_.close().ignore
)
/** Create a stub backend for testing, which uses the [[Task]] response wrapper, and supports `Stream[Throwable,
* ByteBuffer]` streaming.
*
* See [[WebSocketStreamBackendStub]] for details on how to configure stub responses.
*/
def stub: WebSocketStreamBackendStub[Task, ZioStreams] = WebSocketStreamBackendStub(new RIOMonadAsyncError[Any])
val stubLayer: Layer[Throwable, SttpClient with SttpClientStubbing] = SttpClientStubbing.layer
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy