Akka:如何将消息内容包装到HTTP响应中?

时间:2016-07-12 11:48:33

标签: scala akka akka-http

在我的Akka-http路由中,我收到一条特定的消息,我想将其内容包装为错误消息,如:

val response:Future[T] = (actor ? command).mapTo[T]
    response match {
      case err : Future[InvalidRequest] => 
          HttpResponse(408, entity = err.map(_.toJson).????)
      case r : Future[T] => r.map(_.toJson)
    }

case class InvalidRequest(error:String)

implicit val invalidRequestFormat = jsonFormat1(InvalidRequest)

但这不起作用。如何以json格式将其映射为文本?

1 个答案:

答案 0 :(得分:2)

我想我可以提供一个通用的解决方案来解决你想要做的事情。您可以首先创建一个返回Route的方法,如下所示:

def service[T:ClassTag](actor:ActorRef, command:Any)
 (implicit timeout:Timeout,  _marshaller: ToResponseMarshaller[T]):Route = {
  val fut = (actor ? command).mapTo[ServiceResponse]
  onComplete(fut){
    case util.Success(ir:InvalidRequest) =>
      complete(StatusCodes.BadRequest, ir)

    case util.Success(t:T) =>
      complete(t)

    case util.Failure(ex) =>
      complete(StatusCodes.InternalServerError )        
  }    
}

此方法通过ask向请求的actor发出请求,并获取表示结果的Future。然后,它使用onComplete指令对InvalidResponse案例应用特殊处理。这里重要的是你在范围内有一个隐含的ToResponseMarshaller[T],因为你需要成功案例。

然后,假设您已定义以下类和格式化程序:

trait ServiceResponse
case class Foo(id:Int) extends ServiceResponse
implicit val fooFormat = jsonFormat1(Foo)
case class InvalidRequest(error:String) extends ServiceResponse
implicit val invalidRequestFormat = jsonFormat1(InvalidRequest)

您可以在路由树中使用新的service方法,如下所示:

val routes:Route = {
  path("api" / "foo"){
    get{
      service[Foo](fooActor, FooActor.DoFoo)
    }
  }  
}

您的示例的问题在于您在构建响应之前没有等待Future的完成。您试图匹配Future的基础类型,它在运行时被擦除消除,因此以这种方式尝试匹配并不是一个好主意。您需要等到它完成后再看到Future后面的类型。