Scala,Akka,Spray:如何在处理之前验证json数据?

时间:2015-02-01 18:45:36

标签: json validation scala akka spray-json

当所有输入都有效时,我可以处理这个json,即有效键(包括大小写)和值。下一步是验证密钥并在密钥或值无效时返回400(错误请求)。添加此验证的好方法是什么?

API调用

POST http://localhost:8080/api/v1/adsession
Content-Type: application/json
body {
  "sessionId": "abcd123123123",
  "serviceGroup": "1234",
  "targetCode": {"zipcodes":"30096,30188","code2":"value2"}
}

路线处理程序

class AdSessionRoutes(services: Services)(implicit ec: ExecutionContext, log: LoggingContext) extends ApiRoute(services) {

  implicit val timeout = Timeout(10 seconds)

  val postSession = pathPrefix("adsession") & pathEnd & post

  val route: Route = {
    withService("adSession") { service =>

      postSession {
        entity(as[AdSession]) { adSession =>
          log.info(s"Processing POST ${adSession}")
          val future = (service ? CreateAdSession(adSession)).mapTo[AdSession]

          onComplete(future) {
            case Success(result) =>
              complete(StatusCodes.Created, result)

            case Failure(e) =>
              log.error(s"Error: ${e.toString}")
              complete(StatusCodes.InternalServerError, Message(ApiMessages.UnknownException))
          }
        }
      }
    }
  }
}

模型对象

case class AdSession(
  sessionId: String,
  serviceGroup: String,
  targetCodes: Map[String,String],
  id: Option[String] = None)

object AdSessionJsonProtocol extends DefaultJsonProtocol {
  implicit val adSessionFormat = jsonFormat4(AdSession)
}

entity(作为[AdSession])将键映射到case类字段,但我不确定如何捕获这些错误。我想捕获这些错误以及添加其他验证并返回400并使用有效的json错误响应。

2 个答案:

答案 0 :(得分:2)

read定义您自己的writeAdSession方法,如下所示:

object AdSessionJsonProtocol {
    implicit object adSessionJsonFormat extends RootJsonFormat[AdSession] {
        override def read(json: JsValue): AdSession = ???
        override def write(obj: AdSession): JsValue = ???
    }
}

read内部,您可以执行其他验证。

另一个问题是如何将收集的错误传递给Spray路线。我建议你将AdSession包裹到Either[String, AdSession]中,例如:

postSession {
        entity(as[Either[String, AdSession]]) { adSession =>

所以,现在你的adSessionJsonFormat将会是:

implicit object adSessionJsonFormat extends RootJsonFormat[Either[String, AdSession]] {

    override def read(json: JsValue): Either[String, AdSession] = {
      // json.asJsObject.fields access fields, perform checks
      // if everything is OK return Right(AdSession(...))
      // otherwise return Lift("Error Message")
    }

    override def write(obj: Either[String, AdSession]): JsValue = ???
}

但是,我认为可以使用一些隐含的魔法以更优雅的方式解决它。

答案 1 :(得分:2)

我知道这可能有点晚了但是从akka-http-2.4.6开始,你可以将验证逻辑放在case类中。查看this了解如何操作的信息。