com.rojoma.json.v3.zipper.JsonZipper.scala Maven / Gradle / Ivy
package com.rojoma.json.v3
package zipper
import scala.language.implicitConversions
import scala.annotation.tailrec
import ast._
import codec.Path
/** A common parent representing both [[com.rojoma.json.v3.zipper.JsonZipper]]s and the
* [[com.rojoma.json.v3.zipper.NothingZipper]]s which result from removing items from
* the tree of [[com.rojoma.json.v3.ast.JValue]]s. */
sealed trait ZipperLike {
/** Move to the parent object.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the parent object,
* or `None` if this is the top-level object. */
def up: Option[JsonZipper]
/** Move to the parent object.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the parent object.
* @throws NoSuchElementException if this is the top-level object. */
def up_! : JsonZipper
/** Move to the next element in the parent array.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the next element,
* or `None` if there is no next element or if the parent is not a
* [[com.rojoma.json.v3.ast.JArray]]. */
def next: Option[JsonZipper]
/** Move to the previous element in the parent array.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the previous element,
* or `None` if there is no previous element or if the parent is not a
* [[com.rojoma.json.v3.ast.JArray]]. */
def prev: Option[JsonZipper]
/** Move to the next element in the parent array.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the next element.
* @throws NoSuchElementException if there is no next element or if the parent is not
* a [[com.rojoma.json.v3.ast.JArray]]. */
def next_! = next.getOrElse(throw new NoSuchElementException("No next sibling"))
/** Move to the previous element in the parent array.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the previous element.
* @throws NoSuchElementException if there is no previous element or if the parent is not
* a [[com.rojoma.json.v3.ast.JArray]]. */
def prev_! = prev.getOrElse(throw new NoSuchElementException("No previous sibling"))
/** Move to a different field in the parent object.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the new field,
* or `None` if that field does not exist or the parent is not a
* [[com.rojoma.json.v3.ast.JObject]]. */
def sibling(field: String): Option[JsonZipper]
/** Move to a different field in the parent object.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the new field.
* @throws NoSuchElementException if there is no such field or if the parent is not a
* [[com.rojoma.json.v3.ast.JObject]]. */
def sibling_!(field: String) = sibling(field).getOrElse(throw new NoSuchElementException("No sibling " + field))
/** Replace the current value with an atom.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the same location but
* with the current value replaced. */
def replace(newValue: JAtom): JAtomZipper
/** Replace the current value with an array.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the same location but
* with the current value replaced. */
def replace(newValue: JArray): JArrayZipper
/** Replace the current value with an object.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the same location but
* with the current value replaced. */
def replace(newValue: JObject): JObjectZipper
/** Replace the current value with a new value.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the same location but
* with the current value replaced. */
def replace(newValue: JValue): JsonZipper = newValue match {
case atom: JAtom => replace(atom)
case array: JArray => replace(array)
case obj: JObject => replace(obj)
}
private[zipper] def reversePath: List[Path.Entry]
def path: Path = new Path(reversePath.reverse)
}
/** A zipper referencing the hole left after a `remove` operation. */
sealed trait NothingZipper extends ZipperLike {
/** Move up the chain of parents to the top of the object.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the top object,
* or `None` if the top object was removed. */
def top : Option[JsonZipper]
/** Move up the chain of parents to the top of the object.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the top object.
* @throws NoSuchElementException if the top object was removed. */
def top_! : JsonZipper
}
/** A zipper that points somewhere in the tree defined by a [[com.rojoma.json.v3.ast.JValue]].
* It can be used to move around or update the tree in a purely functional manner.
*
* @see [[com.rojoma.json.v3.jpath.JPath]] for a higher-level read-only interface to this
* functionality. */
sealed trait JsonZipper extends ZipperLike {
type ValueType <: JValue
/** The value stored at this location */
def value: ValueType
/** Move up the chain of parents to the top of the object.
*
* @return A [[com.rojoma.json.v3.zipper.JsonZipper]] pointing at the top object. */
def top : JsonZipper
/** Remove the current value from the tree.
*
* @return A [[com.rojoma.json.v3.zipper.NothingZipper]] pointing at the hole
* left by removing the current value. */
def remove: NothingZipper
/** Safe downcast to [[com.rojoma.json.v3.zipper.JAtomZipper]] */
def asAtom = this.cast[JAtomZipper]
/** Safe downcast to [[com.rojoma.json.v3.zipper.JArrayZipper]] */
def asArray = this.cast[JArrayZipper]
/** Safe downcast to [[com.rojoma.json.v3.zipper.JObjectZipper]] */
def asObject = this.cast[JObjectZipper]
def adjust(f: ValueType => JValue): JsonZipper = replace(f(value))
}
/** A [[com.rojoma.json.v3.zipper.JsonZipper]] that points to a [[com.rojoma.json.v3.ast.JAtom]]. */
sealed trait JAtomZipper extends JsonZipper {
type ValueType = JAtom
}
object JAtomZipper {
def apply(value: JAtom) = new TopLevelAtomZipper(value)
def unapply(zipper: JsonZipper) = zipper.cast[JAtomZipper].map(_.value)
}
/** A [[com.rojoma.json.v3.zipper.JsonZipper]] that points to a [[com.rojoma.json.v3.ast.JArray]]. */
sealed trait JArrayZipper extends JsonZipper {
type ValueType = JArray
private def subZipper(elem: JValue, idx: Int): JsonZipper = elem match {
case atom: JAtom => new UnchangedArrayElementAtomZipper(atom, this, idx)
case array: JArray => new UnchangedArrayElementArrayZipper(array, this, idx)
case obj: JObject => new UnchangedArrayElementObjectZipper(obj, this, idx)
}
def down(idx: Int): Option[JsonZipper] =
if(idx < 0 || idx >= length) None
else Some(subZipper(value(idx), idx))
def down_!(idx: Int): JsonZipper =
subZipper(value(idx), idx)
def first = down(0)
def first_! = down_!(0)
def last = down(size - 1)
def last_! = down_!(size - 1)
def remove(idx: Int) = down(idx) match {
case Some(child) => child.remove.up_!.asInstanceOf[JArrayZipper]
case None => None
}
def map(f: JValue => JValue): JArrayZipper
def collect(f: PartialFunction[JValue, JValue]): JArrayZipper
def set(idx: Int, value: JValue): JArrayZipper = down_!(idx).replace(value).up_!.asInstanceOf[JArrayZipper]
def :+(value: JValue): JArrayZipper
def +:(value: JValue): JArrayZipper
def isEmpty = value.isEmpty
def nonEmpty = value.nonEmpty
lazy val size = value.size
def length = size
}
object JArrayZipper {
def apply(value: JArray): JArrayZipper = new TopLevelArrayZipper(value)
def unapply(zipper: JsonZipper) = zipper.cast[JArrayZipper].map(_.value)
}
/** A [[com.rojoma.json.v3.zipper.JsonZipper]] that points to a [[com.rojoma.json.v3.ast.JObject]]. */
sealed trait JObjectZipper extends JsonZipper {
type ValueType = JObject
private def subZipper(elem: JValue, field: String): JsonZipper = elem match {
case atom: JAtom => new UnchangedObjectElementAtomZipper(atom, this, field)
case array: JArray => new UnchangedObjectElementArrayZipper(array, this, field)
case obj: JObject => new UnchangedObjectElementObjectZipper(obj, this, field)
}
def down(field: String): Option[JsonZipper] = value.get(field) match {
case Some(elem) => Some(subZipper(elem, field))
case None => None
}
def down_!(field: String): JsonZipper = subZipper(value(field), field)
def remove(field: String): JObjectZipper =
down(field) match {
case Some(child) => child.remove.up_!.asInstanceOf[JObjectZipper]
case None => this
}
def set(field: String, value: JValue): JObjectZipper
def map(f: ((String, JValue)) => JValue): JObjectZipper
def contains(f: String) = value.contains(f)
}
object JObjectZipper {
def apply(value: JObject): JObjectZipper = new TopLevelObjectZipper(value)
def unapply(zipper: JsonZipper) = zipper.cast[JObjectZipper].map(_.value)
}
object JsonZipper {
def apply(value: JAtom) = JAtomZipper(value)
def apply(value: JArray) = JArrayZipper(value)
def apply(value: JObject) = JObjectZipper(value)
def apply(value: JValue): JsonZipper = value match {
case atom: JAtom => apply(atom)
case array: JArray => apply(array)
case obj: JObject => apply(obj)
}
def unapply(zipper: JsonZipper): Option[JValue] = Some(zipper.value)
@deprecated(message = "Do not use this function", since = "3.14.1")
def toCastable[T <: JsonZipper](x: T): com.rojoma.`json-impl`.DownCaster[T] = new com.rojoma.`json-impl`.DownCaster(x)
implicit def toCorrectCastable[T <: JsonZipper](x: T): com.rojoma.json.v3.`-impl`.GenericDownCaster[T] = new com.rojoma.json.v3.`-impl`.GenericDownCaster(x)
}
// TOP LEVEL
private[zipper] class TopLevelZipperLike { this: ZipperLike =>
def up = None
def up_! = throw new NoSuchElementException("Cannot go up from the top level")
def next = None
def prev = None
def sibling(field: String) = None
def replace(newValue: JAtom) = JAtomZipper(newValue)
def replace(newValue: JArray) = JArrayZipper(newValue)
def replace(newValue: JObject) = JObjectZipper(newValue)
private[zipper] def reversePath = Nil
}
private[zipper] object TopLevelNothingZipper extends TopLevelZipperLike with NothingZipper {
def top = None
def top_! = throw new NoSuchElementException("No top-level object")
}
private[zipper] class TopLevelZipper extends TopLevelZipperLike { this: JsonZipper =>
def remove = TopLevelNothingZipper
def top : JsonZipper = this
}
private[zipper] class TopLevelAtomZipper(val value: JAtom) extends TopLevelZipper with JAtomZipper
private[zipper] class TopLevelArrayZipper(val value: JArray) extends TopLevelZipper with JArrayZipper {
def :+(newValue: JValue): JArrayZipper = new TopLevelArrayZipper(JArray(value.toSeq :+ newValue))
def +:(newValue: JValue): JArrayZipper = new TopLevelArrayZipper(JArray(newValue +: value.toSeq))
def map(f: JValue => JValue) = new TopLevelArrayZipper(JArray(value.toSeq.map(f)))
def collect(f: PartialFunction[JValue, JValue]) = new TopLevelArrayZipper(JArray(value.toSeq.collect(f)))
}
private[zipper] class TopLevelObjectZipper(val value: JObject) extends TopLevelZipper with JObjectZipper {
def set(newField: String, newValue: JValue): JObjectZipper = new TopLevelObjectZipper(JObject(value.fields + (newField -> newValue)))
def map(f: ((String, JValue)) => JValue) = new TopLevelObjectZipper(JObject(value.fields.map { kv => (kv._1, f(kv)) }))
}
// CODE COMMON TO ALL "ELEMENT" ZIPPER(LIKE)S
private[zipper] abstract class ElementZipperLike[Parent <: JsonZipper](parent : Parent) { this: ZipperLike =>
def up_! = parent
def up = Some(up_!)
}
private[zipper] trait ElementZipper[Parent <: JsonZipper] { this: ElementZipperLike[Parent] with JsonZipper =>
def top = {
@tailrec
def loop(last: JsonZipper, next: Option[JsonZipper]): JsonZipper = next match {
case None => last
case Some(p) => loop(p, p.up)
}
loop(this, up)
}
}
private[zipper] trait ElementNothingZipper[Parent <: JsonZipper] { this: ElementZipperLike[Parent] with NothingZipper =>
def top_! : JsonZipper = up_!.top
def top = Some(top_!)
}
// ARRAY ELEMENT, CHANGED OR NOT
private[zipper] abstract class ArrayElementZipperLike(p: JArrayZipper, val idxInParent: Int) extends ElementZipperLike(p) { this: ZipperLike =>
def replace(newValue: JAtom) = new ChangedArrayElementAtomZipper(newValue, up_!, idxInParent)
def replace(newValue: JArray) = new ChangedArrayElementArrayZipper(newValue, up_!, idxInParent)
def replace(newValue: JObject) = new ChangedArrayElementObjectZipper(newValue, up_!, idxInParent)
def remove = new ChangedArrayElementNothingZipper(up_!, idxInParent)
def next : Option[JsonZipper] = up_!.down(idxInParent + 1)
def prev : Option[JsonZipper] = up_!.down(idxInParent - 1)
def sibling(field: String) = None
private[zipper] def reversePath = Path.Index(idxInParent) :: p.reversePath
}
private[zipper] abstract class ArrayElementZipper(p: JArrayZipper, i: Int) extends ArrayElementZipperLike(p, i) with ElementZipper[JArrayZipper] { this: JsonZipper =>
}
private[zipper] trait ArrayElementArrayZipper { this: ArrayElementZipper with JArrayZipper =>
def :+(newValue: JValue): JArrayZipper = new ChangedArrayElementArrayZipper(JArray(value.toSeq :+ newValue), up_!, idxInParent)
def +:(newValue: JValue): JArrayZipper = new ChangedArrayElementArrayZipper(JArray(newValue +: value.toSeq), up_!, idxInParent)
def map(f: JValue => JValue) = new ChangedArrayElementArrayZipper(JArray(value.toSeq.map(f)), up_!, idxInParent)
def collect(f: PartialFunction[JValue, JValue]) = new ChangedArrayElementArrayZipper(JArray(value.toSeq.collect(f)), up_!, idxInParent)
}
private[zipper] trait ArrayElementObjectZipper { this: ArrayElementZipper with JObjectZipper =>
def set(newField: String, newValue: JValue): JObjectZipper = new ChangedArrayElementObjectZipper(JObject(value.fields + (newField -> newValue)), up_!, idxInParent)
def map(f: ((String, JValue)) => JValue) = new ChangedArrayElementObjectZipper(JObject(value.fields.map { kv => (kv._1, f(kv)) }), up_!, idxInParent)
}
// OBJECT ELEMENT, CHANGED OR NOT
private[zipper] abstract class ObjectElementZipperLike(p: JObjectZipper, val fieldInParent: String) extends ElementZipperLike(p) { this: ZipperLike =>
def replace(newValue: JAtom) = new ChangedObjectElementAtomZipper(newValue, up_!, fieldInParent)
def replace(newValue: JArray) = new ChangedObjectElementArrayZipper(newValue, up_!, fieldInParent)
def replace(newValue: JObject) = new ChangedObjectElementObjectZipper(newValue, up_!, fieldInParent)
def remove = new ChangedObjectElementNothingZipper(up_!, fieldInParent)
def next = None
def prev = None
def sibling(field: String): Option[JsonZipper] = up_!.down(field)
private[zipper] def reversePath = Path.Field(fieldInParent) :: p.reversePath
}
private[zipper] abstract class ObjectElementZipper(p: JObjectZipper, f: String) extends ObjectElementZipperLike(p, f) with ElementZipper[JObjectZipper] { this: JsonZipper =>
}
private[zipper] trait ObjectElementArrayZipper { this: ObjectElementZipper with JArrayZipper =>
def :+(newValue: JValue): JArrayZipper = new ChangedObjectElementArrayZipper(JArray(value.toSeq :+ newValue), up_!, fieldInParent)
def +:(newValue: JValue): JArrayZipper = new ChangedObjectElementArrayZipper(JArray(newValue +: value.toSeq), up_!, fieldInParent)
def map(f: JValue => JValue) = new ChangedObjectElementArrayZipper(JArray(value.toSeq.map(f)), up_!, fieldInParent)
def collect(f: PartialFunction[JValue, JValue]) = new ChangedObjectElementArrayZipper(JArray(value.toSeq.collect(f)), up_!, fieldInParent)
}
private[zipper] trait ObjectElementObjectZipper { this: ObjectElementZipper with JObjectZipper =>
def set(newField: String, newValue: JValue): JObjectZipper = new ChangedObjectElementObjectZipper(JObject(value.fields + (newField -> newValue)), up_!, fieldInParent)
def map(f: ((String, JValue)) => JValue) = new ChangedObjectElementObjectZipper(JObject(value.fields.map { kv => (kv._1, f(kv)) }), up_!, fieldInParent)
}
// ARRAY ELEMENT BUT UNCHANGED
private[zipper] abstract class UnchangedArrayElementZipper(p: JArrayZipper, i: Int) extends ArrayElementZipper(p, i) { this: JsonZipper =>
}
private[zipper] class UnchangedArrayElementAtomZipper(val value: JAtom, p: JArrayZipper, i: Int) extends UnchangedArrayElementZipper(p, i) with JAtomZipper
private[zipper] class UnchangedArrayElementArrayZipper(val value: JArray, p: JArrayZipper, i: Int) extends UnchangedArrayElementZipper(p, i) with ArrayElementArrayZipper with JArrayZipper
private[zipper] class UnchangedArrayElementObjectZipper(val value: JObject, p: JArrayZipper, i: Int) extends UnchangedArrayElementZipper(p, i) with ArrayElementObjectZipper with JObjectZipper
// OBJECT ELEMENT BUT UNCHANGED
private[zipper] abstract class UnchangedObjectElementZipper(p: JObjectZipper, f: String) extends ObjectElementZipper(p, f) { this: JsonZipper =>
}
private[zipper] class UnchangedObjectElementAtomZipper(val value: JAtom, p: JObjectZipper, f: String) extends UnchangedObjectElementZipper(p, f) with JAtomZipper
private[zipper] class UnchangedObjectElementArrayZipper(val value: JArray, p: JObjectZipper, f: String) extends UnchangedObjectElementZipper(p, f) with ObjectElementArrayZipper with JArrayZipper
private[zipper] class UnchangedObjectElementObjectZipper(val value: JObject, p: JObjectZipper, f: String) extends UnchangedObjectElementZipper(p, f) with ObjectElementObjectZipper with JObjectZipper
// ARRAY ELEMENT AND CHANGED
private[zipper] abstract class ChangedArrayElementZipper(p: JArrayZipper, i: Int) extends ArrayElementZipper(p, i) { this: JsonZipper =>
override def up_! = super.up_!.replace(JArray(super.up_!.value.toSeq.updated(idxInParent, value)))
}
private[zipper] class ChangedArrayElementAtomZipper(val value: JAtom, p: JArrayZipper, i: Int) extends ChangedArrayElementZipper(p, i) with JAtomZipper
private[zipper] class ChangedArrayElementArrayZipper(val value: JArray, p: JArrayZipper, i: Int) extends ChangedArrayElementZipper(p, i) with ArrayElementArrayZipper with JArrayZipper
private[zipper] class ChangedArrayElementObjectZipper(val value: JObject, p: JArrayZipper, i: Int) extends ChangedArrayElementZipper(p, i) with ArrayElementObjectZipper with JObjectZipper
private[zipper] class ChangedArrayElementNothingZipper(parent: JArrayZipper, i: Int) extends ArrayElementZipperLike(parent, i) with ElementNothingZipper[JArrayZipper] with NothingZipper {
override def up_! = super.up_!.replace(JArray(super.up_!.value.toSeq.patch(idxInParent, Nil, 1)))
override def replace(newValue: JAtom) = new ChangedArrayElementAtomZipper(newValue, parent, idxInParent)
override def replace(newValue: JArray) = new ChangedArrayElementArrayZipper(newValue, parent, idxInParent)
override def replace(newValue: JObject) = new ChangedArrayElementObjectZipper(newValue, parent, idxInParent)
override def remove = this
override def next : Option[JsonZipper] = up_!.down(idxInParent)
}
// OBJECT ELEMENT AND CHANGED
private[zipper] abstract class ChangedObjectElementZipper(p: JObjectZipper, f: String) extends ObjectElementZipper(p, f) with ElementZipper[JObjectZipper] { this: JsonZipper =>
override def up_! = super.up_!.replace(JObject(super.up_!.value.fields.updated(fieldInParent, value)))
}
private[zipper] class ChangedObjectElementAtomZipper(val value: JAtom, p: JObjectZipper, f: String) extends ChangedObjectElementZipper(p, f) with JAtomZipper
private[zipper] class ChangedObjectElementArrayZipper(val value: JArray, p: JObjectZipper, f: String) extends ChangedObjectElementZipper(p, f) with ObjectElementArrayZipper with JArrayZipper
private[zipper] class ChangedObjectElementObjectZipper(val value: JObject, p: JObjectZipper, f: String) extends ChangedObjectElementZipper(p, f) with ObjectElementObjectZipper with JObjectZipper
private[zipper] class ChangedObjectElementNothingZipper(parent: JObjectZipper, field: String) extends ObjectElementZipperLike(parent, field) with ElementNothingZipper[JObjectZipper] with NothingZipper {
override def up_! = super.up_!.replace(JObject(super.up_!.value.fields - field))
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy