在Play中将Json转换为Joda DateTime!斯卡拉2.5

时间:2016-11-03 16:15:26

标签: json scala datetime playframework jodatime

我已经阅读了这个question(以及SO上的其他内容),但我仍然无法使用Joda DateTime将JsValue转换为案例类。

这是我的JSON:

val json = Json.parse(
    """
    {
        "message": "Bla bla bla",
        "updated_time": "2016-09-17T12:48:12+0000"
    }
    """
)

相应的案例类是:

import org.joda.time.DateTime
final case class FacebookPost(message: String, updated_time: DateTime)

我还添加了这个隐式Da​​teTime阅读器(我也尝试了其他一些):

 implicit val readsJodaLocalDateTime = Reads[DateTime](js =>
        js.validate[String].map[DateTime](dtString => new DateTime(dtString)))

但是当我尝试将我的json转换为相应的case类(json.as[FacebookPost])时,我收到此错误:

play.api.libs.json.JsResultException: `JsResultException(errors:List((/updated_time,List(ValidationError(List(error.expected.jodadate.format),WrappedArray(yyyy-MM-dd))))))`

我做错了什么?

1 个答案:

答案 0 :(得分:3)

您的问题是日期时间的格式:2016-09-17T12:48:12+0000

默认情况下,Joda的DateTime课程无法自动确定您使用的格式,因此您必须使用DateTimeFormat提供提示。

自定义Reads[DateTime]的示例:

import play.api.libs.json._
import play.api.libs.json.Reads._
import play.api.libs.functional.syntax._ 

val dateFormat = "yyyy-MM-dd'T'HH:mm:ssZ"

val customReads = Reads[DateTime](js =>
  js.validate[String].map[DateTime](dtString =>
    DateTime.parse(dtString, DateTimeFormat.forPattern(dateFormat))
  )
)

implicit val reads: Reads[FacebookPost] = (
  (__ \ "message").read[String] and
  (__ \ "updated_time").read[DateTime](customReads)
)(FacebookPost.apply _)

然后,可以执行以下操作:

val json = Json.parse(
  """
    {
      "message": "Bla bla bla",
      "updated_time": "2016-09-17T12:48:12+0000"
    }
   """)

val post = json.validate[FacebookPost] 
// => JsSuccess(FacebookPost(Bla bla bla,2016-09-17T14:48:12.000+02:00))

修改

您无需从头开始创建Read[DateTime],Play中有一个预定义的方法可以帮助您:

object CustomReads /* or wherever you want to define it */ {
  val dateFormat = "yyyy-MM-dd'T'HH:mm:ssZ"
  implicit val jodaDateReads = play.api.libs.json.Reads.jodaDateReads(dateFormat)
}

完成。只需在需要时导入,例如在模型类的伴随对象中:

object FacebookPost {
  import utils.CustomReads._
  implicit val reads: Reads[FacebookPost] = Json.reads[FacebookPost]
}

您可以通过从导入中排除来“覆盖”Play提供的隐式Read[DateTime]

import play.api.libs.json.Reads.{DefaultJodaDateReads => _, _}