org.json4sbt.MonadicJValue.scala Maven / Gradle / Ivy
The newest version!
package org.json4sbt
import java.util.Locale.ENGLISH
class MonadicJValue(jv: JValue) {
/**
* XPath-like expression to query JSON fields by name. Matches only fields on
* next level.
*
* Example:
* json \ "name"
*
*/
def \(nameToFind: String): JValue = jv match {
case JArray(xs) => JArray(findDirectByName(xs, nameToFind))
case _ =>
findDirectByName(List(jv), nameToFind) match {
case Nil ⇒ JNothing
case x :: Nil ⇒ x
case x ⇒ JArray(x)
}
}
private[this] def findDirectByName(xs: List[JValue], name: String): List[JValue] = xs.flatMap {
case JObject(l) ⇒ l.filter {
case (n, _) if n == name ⇒ true
case _ ⇒ false
} map (_._2)
case JArray(l) ⇒ findDirectByName(l, name)
case _ ⇒ Nil
}
private[this] def findDirect(xs: List[JValue], p: JValue ⇒ Boolean): List[JValue] = xs.flatMap {
case JObject(l) ⇒ l.filter {
case (n, x) if p(x) ⇒ true
case _ ⇒ false
} map (_._2)
case JArray(l) ⇒ findDirect(l, p)
case x if p(x) ⇒ x :: Nil
case _ ⇒ Nil
}
/**
* XPath-like expression to query JSON fields by name. Returns all matching fields.
*
* Example:
* json \\ "name"
*
*/
def \\(nameToFind: String): JValue = {
def find(json: JValue): List[JField] = json match {
case JObject(l) ⇒ l.foldLeft(List[JField]()) {
case (a, (name, value)) ⇒
if (name == nameToFind) a ::: List((name, value)) ::: find(value) else a ::: find(value)
}
case JArray(l) ⇒ l.foldLeft(List[JField]())((a, json) ⇒ a ::: find(json))
case _ ⇒ Nil
}
find(jv) match {
case (_, x) :: Nil ⇒ x
case xs ⇒ JObject(xs)
}
}
/**
* XPath-like expression to query JSON fields by type. Matches only fields on
* next level.
*
* Example:
* json \ classOf[JInt]
*
*/
def \[A <: JValue](clazz: Class[A]): List[A#Values] =
findDirect(jv.children, typePredicate(clazz) _).asInstanceOf[List[A]] map { _.values }
/**
* XPath-like expression to query JSON fields by type. Returns all matching fields.
*
* Example:
* json \\ classOf[JInt]
*
*/
def \\[A <: JValue](clazz: Class[A]): List[A#Values] =
(jv filter typePredicate(clazz) _).asInstanceOf[List[A]] map { _.values }
private def typePredicate[A <: JValue](clazz: Class[A])(json: JValue) = json match {
case x if x.getClass == clazz ⇒ true
case _ ⇒ false
}
/**
* Return a combined value by folding over JSON by applying a function f
* for each element. The initial value is z
.
*/
def fold[A](z: A)(f: (A, JValue) ⇒ A): A = {
def rec(acc: A, v: JValue) = {
val newAcc = f(acc, v)
v match {
case JObject(l) ⇒ l.foldLeft(newAcc) { case (a, (name, value)) ⇒ value.fold(a)(f) }
case JArray(l) ⇒ l.foldLeft(newAcc)((a, e) ⇒ e.fold(a)(f))
case _ ⇒ newAcc
}
}
rec(z, jv)
}
/**
* Return a combined value by folding over JSON by applying a function f
* for each field. The initial value is z
.
*/
def foldField[A](z: A)(f: (A, JField) ⇒ A): A = {
def rec(acc: A, v: JValue) = {
v match {
case JObject(l) ⇒ l.foldLeft(acc) {
case (a, field @ (name, value)) ⇒ value.foldField(f(a, field))(f)
}
case JArray(l) ⇒ l.foldLeft(acc)((a, e) ⇒ e.foldField(a)(f))
case _ ⇒ acc
}
}
rec(z, jv)
}
/**
* Return a new JValue resulting from applying the given function f
* to each value in JSON.
*
* Example:
* JArray(JInt(1) :: JInt(2) :: Nil) map {
* case JInt(x) => JInt(x+1)
* case x => x
* }
*
*/
def map(f: JValue ⇒ JValue): JValue = {
def rec(v: JValue): JValue = v match {
case JObject(l) ⇒ f(JObject(l.map { case (n, va) ⇒ (n, rec(va)) }))
case JArray(l) ⇒ f(JArray(l.map(rec)))
case x ⇒ f(x)
}
rec(jv)
}
/**
* Return a new JValue resulting from applying the given function f
* to each field in JSON.
*
* Example:
* JObject(("age", JInt(10)) :: Nil) mapField {
* case ("age", JInt(x)) => ("age", JInt(x+1))
* case x => x
* }
*
*/
def mapField(f: JField ⇒ JField): JValue = {
def rec(v: JValue): JValue = v match {
case JObject(l) ⇒ JObject(l.map { case (n, va) ⇒ f(n -> rec(va)) })
case JArray(l) ⇒ JArray(l.map(rec))
case x ⇒ x
}
rec(jv)
}
/**
* Return a new JValue resulting from applying the given partial function f
* to each field in JSON.
*
* Example:
* JObject(("age", JInt(10)) :: Nil) transformField {
* case ("age", JInt(x)) => ("age", JInt(x+1))
* }
*
*/
def transformField(f: PartialFunction[JField, JField]): JValue = mapField { x ⇒
if (f.isDefinedAt(x)) f(x) else x
}
/**
* Return a new JValue resulting from applying the given partial function f
* to each value in JSON.
*
* Example:
* JArray(JInt(1) :: JInt(2) :: Nil) transform { case JInt(x) => JInt(x+1) }
*
*/
def transform(f: PartialFunction[JValue, JValue]): JValue = map { x ⇒
if (f.isDefinedAt(x)) f(x) else x
}
/**
* Return a new JValue resulting from replacing the value at the specified field
* path with the replacement value provided. This has no effect if the path is empty
* or if the value is not a JObject instance.
*
* Example:
* JObject(List(JField("foo", JObject(List(JField("bar", JInt(1))))))).replace("foo" :: "bar" :: Nil, JString("baz"))
* // returns JObject(List(JField("foo", JObject(List(JField("bar", JString("baz")))))))
*
*/
def replace(l: List[String], replacement: JValue): JValue = {
def rep(l: List[String], in: JValue): JValue = {
l match {
case x :: xs ⇒ in match {
case JObject(fields) ⇒ JObject(
fields.map {
case JField(`x`, value) ⇒ JField(x, if (xs == Nil) replacement else rep(xs, value))
case field ⇒ field
})
case other ⇒ other
}
case Nil ⇒ in
}
}
rep(l, jv)
}
/**
* Return the first field from JSON which matches the given predicate.
*
* Example:
* JObject(("age", JInt(2))) findField { case (n, v) => n == "age" }
*
*/
def findField(p: JField ⇒ Boolean): Option[JField] = {
def find(json: JValue): Option[JField] = json match {
case JObject(fs) if (fs find p).isDefined ⇒ fs find p
case JObject(fs) ⇒ fs.flatMap { case (n, v) ⇒ find(v) }.headOption
case JArray(l) ⇒ l.flatMap(find _).headOption
case _ ⇒ None
}
find(jv)
}
/**
* Return the first element from JSON which matches the given predicate.
*
* Example:
* JArray(JInt(1) :: JInt(2) :: Nil) find { _ == JInt(2) } == Some(JInt(2))
*
*/
def find(p: JValue ⇒ Boolean): Option[JValue] = {
def find(json: JValue): Option[JValue] = {
if (p(json)) return Some(json)
json match {
case JObject(fs) ⇒ fs.flatMap { case (n, v) ⇒ find(v) }.headOption
case JArray(l) ⇒ l.flatMap(find _).headOption
case _ ⇒ None
}
}
find(jv)
}
/**
* Return a List of all fields which matches the given predicate.
*
* Example:
* JObject(("age", JInt(10)) :: Nil) filterField {
* case ("age", JInt(x)) if x > 18 => true
* case _ => false
* }
*
*/
def filterField(p: JField ⇒ Boolean): List[JField] =
foldField(List[JField]())((acc, e) ⇒ if (p(e)) e :: acc else acc).reverse
/**
* Return a List of all values which matches the given predicate.
*
* Example:
* JArray(JInt(1) :: JInt(2) :: Nil) filter { case JInt(x) => x > 1; case _ => false }
*
*/
def filter(p: JValue ⇒ Boolean): List[JValue] =
fold(List[JValue]())((acc, e) ⇒ if (p(e)) e :: acc else acc).reverse
def withFilter(p: JValue => Boolean) = new JValueWithFilter(jv, p)
class JValueWithFilter(self: JValue, p: JValue => Boolean) {
def map[T](f: JValue => T): List[T] =
self.filter(p).map(f)
def flatMap[T](f: JValue => List[T]): List[T] =
self.filter(p).flatMap(f)
def foreach(f: JValue => Unit): Unit =
self.filter(p).foreach(f)
def withFilter(q: JValue => Boolean): JValueWithFilter =
new JValueWithFilter(self, x => p(x) && q(x))
}
/**
* Return a JSON where all fields matching the given predicate are removed.
*
* Example:
* JObject(("age", JInt(10)) :: Nil) removeField {
* case ("age", _) => true
* case _ => false
* }
*
*/
def removeField(p: JField ⇒ Boolean): JValue = jv transform {
case JObject(l) => JObject(l.filterNot(p))
}
/**
* Return a JSON where all values matching the given predicate are removed.
*
* Example:
* JArray(JInt(1) :: JInt(2) :: JNull :: Nil) remove { _ == JNull }
*
*/
def remove(p: JValue ⇒ Boolean): JValue = {
if(p(jv)) JNothing
else jv transform {
case JObject(l) => JObject(l.filterNot(f ⇒ p(f._2)))
case JArray(l) => JArray(l.filterNot(p))
}
}
private[this] def camelize(word: String): String = {
if (word.nonEmpty) {
val w = pascalize(word)
w.substring(0, 1).toLowerCase(ENGLISH) + w.substring(1)
} else {
word
}
}
private[this] def pascalize(word: String): String = {
val lst = word.split("_").toList
(lst.headOption.map(s ⇒ s.substring(0, 1).toUpperCase(ENGLISH) + s.substring(1)).get ::
lst.tail.map(s ⇒ s.substring(0, 1).toUpperCase + s.substring(1))).mkString("")
}
private[this] def underscore(word: String): String = {
val spacesPattern = "[-\\s]".r
val firstPattern = "([A-Z]+)([A-Z][a-z])".r
val secondPattern = "([a-z\\d])([A-Z])".r
val replacementPattern = "$1_$2"
spacesPattern.replaceAllIn(
secondPattern.replaceAllIn(
firstPattern.replaceAllIn(
word, replacementPattern), replacementPattern), "_").toLowerCase
}
/**
* Camelize all the keys in this [[org.json4sbt.JsonAST.JValue]]
*/
def camelizeKeys = rewriteJsonAST(camelize = true)
/**
* Underscore all the keys in this [[org.json4sbt.JsonAST.JValue]]
*/
def snakizeKeys = rewriteJsonAST(camelize = false)
/**
* Underscore all the keys in this [[org.json4sbt.JsonAST.JValue]]
*/
def underscoreKeys = snakizeKeys
private[this] def rewriteJsonAST(camelize: Boolean): JValue =
transformField {
case JField(nm, x) if !nm.startsWith("_") ⇒ JField(if (camelize) this.camelize(nm) else underscore(nm), x)
}
/**
* Remove the [[org.json4sbt.JsonAST.JNothing]] and [[org.json4sbt.JsonAST.JNull]] from
* a [[org.json4sbt.JsonAST.JArray]] or [[org.json4sbt.JsonAST.JObject]]
*/
def noNulls = remove {
case JNull | JNothing => true
case _ => false
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy