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

com.github.matsluni.akkahttpspi.RequestRunner.scala Maven / Gradle / Ivy

Go to download

An alternative non-blocking async http engine for aws-sdk-java-v2 based on akka-http

The newest version!
/*
 * Copyright 2018 Matthias Lüneberg
 *
 * 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 com.github.matsluni.akkahttpspi

import java.util.concurrent.CompletableFuture

import akka.actor.ActorSystem
import akka.http.scaladsl.model.{ContentTypes, HttpResponse}
import akka.http.scaladsl.model.headers.{`Content-Length`, `Content-Type`}
import akka.stream.Materializer
import akka.stream.scaladsl.{Keep, Sink}
import org.slf4j.LoggerFactory
import software.amazon.awssdk.http.SdkHttpFullResponse
import software.amazon.awssdk.http.async.SdkAsyncHttpResponseHandler

import scala.compat.java8.FutureConverters
import scala.concurrent.{ExecutionContext, Future}
import scala.jdk.CollectionConverters._

class RequestRunner()(implicit sys: ActorSystem, ec: ExecutionContext, mat: Materializer) {

  val logger = LoggerFactory.getLogger(this.getClass)

  def run(runRequest: () => Future[HttpResponse],
          handler: SdkAsyncHttpResponseHandler): CompletableFuture[Void] = {
    val result = runRequest().flatMap { response =>
      handler.onHeaders(toSdkHttpFullResponse(response))

      val (complete, publisher) = response
        .entity
        .dataBytes
        .map(_.asByteBuffer)
        .alsoToMat(Sink.ignore)(Keep.right)
        .toMat(Sink.asPublisher(fanout = false))(Keep.both)
        .run()

      handler.onStream(publisher)
      complete
    }

    result.failed.foreach(handler.onError)
    FutureConverters.toJava(result.map(_ => null: Void)).toCompletableFuture
  }

  private[akkahttpspi] def toSdkHttpFullResponse(response: HttpResponse): SdkHttpFullResponse = {
    SdkHttpFullResponse.builder()
      .headers(convertToSdkResponseHeaders(response).map { case (k, v) => k -> v.asJava }.asJava)
      .statusCode(response.status.intValue())
      .statusText(response.status.reason)
      .build
  }

  private[akkahttpspi] def convertToSdkResponseHeaders(response: HttpResponse): Map[String, Seq[String]] = {
    val contentType = response.entity.contentType match {
      case ContentTypes.NoContentType => Map.empty[String, List[String]]
      case contentType => Map(`Content-Type`.name -> List(contentType.value))
    }

    val contentLength = response.entity.contentLengthOption
      .map(length => `Content-Length`.name -> List(length.toString)).toMap

    val headers = response.headers.groupBy(_.name()).map { case (k, v) => k -> v.map(_.value()) }

    headers ++ contentType ++ contentLength
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy