A tool to visually compare Scala data structures with out of the box support for arbitrary case classes.
Be aware: Collections (List, Seq, etc.) are compared like sets, i.e. ignoring order.
"ai.x" %% "diff" % "2.0.1"
Stopped working in 1.2.0 due to what seems like a Scala compiler bug. See #18
import ai.x.diff.DiffShow
import ai.x.diff.conversions._
println( DiffShow.diff[Foo]( before, after ).string )
Be aware that diff throws an Exception if a DiffShow type class instance for some field
can't be found rather than a type error.
If you use diff in a testing or debugging scenario that's usually not a problem.
The advantage is that the Exception can tell exactly which instance wasn't found. A type error
can only point to the outer most class (Foo
in this case) even if it is actually one of it's deeply nested fields that is lacking an instance for it's type. Knowing only Foo
would not be very helpful to pin point
which instance is missing.
sealed trait Parent
case class Bar( s: String, i: Int ) extends Parent
case class Foo( bar: Bar, b: List[Int], parent: Option[Parent] ) extends Parent
val before: Foo = Foo(
Bar( "asdf", 5 ),
List( 123, 1234 ),
Some( Bar( "asdf", 5 ) )
)
val after: Foo = Foo(
Bar( "asdf", 66 ),
List( 1234 ),
Some( Bar( "qwer", 5 ) )
)
import ai.x.diff._
Sometimes you may need to write your own type class instances. For example for non-case classes that don't compare well using ==.
import ai.x.diff._
import org.joda.time.LocalTime
implicit def localTimeDiffShow: DiffShow[LocalTime] = new DiffShow[LocalTime] {
def show ( d: LocalTime ) = "LocalTime(" + d.toString + ")"
def diff( l: LocalTime, r: LocalTime ) =
if ( l isEqual r ) Identical( l ) else Different( l, r )
}
Sometimes you may want to ignore some parts of your data during comparison. You can do so by type, e.g. for non-deterministic parts like ObjectId, which always differ.
def ignore[T]: DiffShow[T] = new DiffShow[T] {
def show( t: T ) = t.toString
def diff( left: T, right: T ) = Identical( "<not compared>" )
override def diffable( left: T, right: T ) = true
}
implicit def LocationIdShow: DiffShow[LocationId] = ignore[LocationId]
When comparing collections you can influence if two elements should be compared or treated as completely different. Comparing elements shows their partial differences. Not comparing them shows them as added or removed.
implicit def PersonDiffShow[L <: HList](
implicit
labelled: LabelledGeneric.Aux[Person, L],
hlistShow: Lazy[DiffShowFields[L]]
): DiffShow[Person] = new CaseClassDiffShow[Person, L] {
override def diffable( left: Person, right: Person ) = left._id === right._id
}