如何通过Circe修改json的值类型

时间:2017-02-02 09:02:08

标签: json scala circe

我需要捕获json的int值,并通过某个映射表将其更改为字符串值。我使用circe,我知道如何在不改变值类型的情况下修改值,例如:

package com.accenture.soh.driver

import io.circe._, io.circe.parser._
import cats.syntax.either._
import io.circe._, io.circe.parser._
import io.circe.optics.JsonPath._

/**
  * Created by laurence.geng on 2017/2/2.
  */
object CirceTest extends App {

  val json: Json = parse(
    """
  [
  {
    "metric":"my-metric",
    "dps":{"1484214238":5,"1484214239":1,"1484214240":4,"1484214241":11}
  }
  ]
  """).getOrElse(Json.Null)

  val doubleValue: Json => Json = root.each.dps.each.int.modify(_ * 2)

  println(doubleValue(json).toString())

}

输出结果为:

[
  {
    "metric" : "my-metric",
    "dps" : {
      "1484214238" : 10,
      "1484214239" : 2,
      "1484214240" : 8,
      "1484214241" : 22
    }
  }
]

但是,实际上,我需要做的是将int值更改为某个字符串值,即1 - > “A”,2 - > “B”,等等,问题是:“modify”方法只接受一个函数,它返回相同类型的输入值,所以,我不能编码如下:

val intToString: Json => Json = root.each.dps.each.int.modify(_.toString)

我的预期输出可能如下所示:

[
  {
    "metric" : "my-metric",
    "dps" : {
      "1484214238" : "10",
      "1484214239" : "2",
      "1484214240" : "8",
      "1484214241" : "22"
    }
  }
]

任何人都可以给我一个workround(基于Circe)吗?

2 个答案:

答案 0 :(得分:4)

您可以使用circe游标执行此操作,但我认为更容易转换为案例类并对其进行操作。

 val json =
  """
    |{
    |    "metric":"my-metric",
    |    "dps":{"1484214238":5,"1484214239":1,"1484214240":4,"1484214241":11}
    |  }
  """.stripMargin

val doc = parse(json).getOrElse(Json.Null)
val cursor = doc.hcursor
val dps = cursor.downField("dps")
val result = dps.withFocus { json: Json =>
  json.mapObject { jsonObject =>
    JsonObject.fromMap(jsonObject.toMap.mapValues { x =>
      Json.fromString(x.asNumber.get.toString)
    })
  }
}

答案 1 :(得分:1)

天真的解决方案:

val doubleValue: Json => Json = root.each.dps.each.json.modify(x =>
  Json.fromString(
    (x.asNumber.getOrElse(throw new Exception("ups")).truncateToInt * 2).toString
  )
)