All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.json4s.Diff.scala Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2009-2010 WorldWide Conferencing, LLC
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.json4s

import JsonAST._

/**
 * A difference between two JSONs (j1 diff j2).
 * @param changed what has changed from j1 to j2
 * @param added what has been added to j2
 * @param deleted what has been deleted from j1
 */
case class Diff(changed: JValue, added: JValue, deleted: JValue) {
  def map(f: JValue => JValue): Diff = {
    def applyTo(x: JValue) = x match {
      case JNothing => JNothing
      case _ => f(x)
    }
    Diff(applyTo(changed), applyTo(added), applyTo(deleted))
  }

  private[json4s] def toField(name: String): Diff = {
    def applyTo(x: JValue) = x match {
      case JNothing => JNothing
      case _ => JObject((name, x))
    }
    Diff(applyTo(changed), applyTo(added), applyTo(deleted))
  }
}

/**
 * Computes a diff between two JSONs.
 */
object Diff {

  /**
   * Return a diff.
   * 

* Example:

   * val Diff(c, a, d) = ("name", "joe") ~ ("age", 10) diff ("fname", "joe") ~ ("age", 11)
   * c = JObject(("age",JInt(11)) :: Nil)
   * a = JObject(("fname",JString("joe")) :: Nil)
   * d = JObject(("name",JString("joe")) :: Nil)
   * 
*/ def diff(val1: JValue, val2: JValue): Diff = (val1, val2) match { case (x, y) if x == y => Diff(JNothing, JNothing, JNothing) case (JObject(xs), JObject(ys)) => diffFields(xs, ys) case (JArray(xs), JArray(ys)) => diffVals(xs, ys) // unlike diff of JArrays, order of elements is ignored in diff of JSets case (xx @ JSet(x), yy @ JSet(y)) if x != y => Diff(JNothing, yy.difference(xx), xx.difference(yy)) case (JInt(x), JInt(y)) if x != y => Diff(JInt(y), JNothing, JNothing) case (JDouble(x), JDouble(y)) if x != y => Diff(JDouble(y), JNothing, JNothing) case (JDecimal(x), JDecimal(y)) if x != y => Diff(JDecimal(y), JNothing, JNothing) case (JString(x), JString(y)) if x != y => Diff(JString(y), JNothing, JNothing) case (JBool(x), JBool(y)) if x != y => Diff(JBool(y), JNothing, JNothing) case (JNothing, x) => Diff(JNothing, x, JNothing) case (x, JNothing) => Diff(JNothing, JNothing, x) case (x @ _, y) => Diff(y, JNothing, JNothing) } private def diffFields(vs1: List[JField], vs2: List[JField]) = { def diffRec(xleft: List[JField], yleft: List[JField]): Diff = xleft match { case Nil => Diff(JNothing, if (yleft.isEmpty) JNothing else JObject(yleft), JNothing) case x :: xs => yleft find (_._1 == x._1) match { case Some(y) => val Diff(c1, a1, d1) = diff(x._2, y._2).toField(y._1) val Diff(c2, a2, d2) = diffRec(xs, yleft filterNot (_ == y)) Diff(c1 merge c2, a1 merge a2, d1 merge d2) case None => val Diff(c, a, d) = diffRec(xs, yleft) Diff(c, a, JObject(x :: Nil) merge d) } } diffRec(vs1, vs2) } private def diffVals(vs1: List[JValue], vs2: List[JValue]) = { def diffRec(xleft: List[JValue], yleft: List[JValue]): Diff = (xleft, yleft) match { case (xs, Nil) => Diff(JNothing, JNothing, if (xs.isEmpty) JNothing else JArray(xs)) case (Nil, ys) => Diff(JNothing, if (ys.isEmpty) JNothing else JArray(ys), JNothing) case (x :: xs, y :: ys) => val Diff(c1, a1, d1) = diff(x, y) val Diff(c2, a2, d2) = diffRec(xs, ys) Diff(c1 ++ c2, a1 ++ a2, d1 ++ d2) } diffRec(vs1, vs2) } private[json4s] trait Diffable { this: org.json4s.JValue => /** * Return a diff. * @see org.json4s.Diff#diff */ def diff(other: JValue): Diff = Diff.diff(this, other) } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy