当所有输入都有效时,我可以处理这个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错误响应。
答案 0 :(得分:2)
为read
定义您自己的write
和AdSession
方法,如下所示:
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了解如何操作的信息。