Skip to content

Javascript lib for producing diffs of data structures and time-traveling with them

License

Notifications You must be signed in to change notification settings

tashburn/diffstory

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

89 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

diffstory

A diff library to assist in managing revision histories of common Javascript data structures.

  • It's small and fast with minimal dependencies
  • Handles objects, arrays, strings, numbers, booleans
  • Can diff from one structure to the other, even if different types
  • Produces small human-readable diffs
  • Diffs are composed of basic non-cyclic Javascript structures that are easily serializable and storable
  • Can use a diff to "time-travel" a structure forward
  • Can use a diff to "time-travel" a structure backwards
  • Uses a fast "smallest common subsequence" algorithm for nice diffs of strings and arrays
  • For objects and arrays, diffs can represent additions, removals, and updates
  • For strings, the diff is a compact string-based format, which can be converted to a processable "list of operations"

Installation

npm i --save diffstory

Basic Usage

import diffstory from 'diffstory'

const obj1 = { a:1, b:2, c:3 }
const obj2 = { b:2, c:30, d:4 }
const diff = diffstory.diff(obj1, obj2) 

// diff is { 
//   '+prop': {d:4}, 
//   '-prop': {a:1}, 
//   '!prop': {c:{'-val':3,'+val':30}}
//   '&prop': {b:2}
// }

Diffing Objects

Additions

const a = { a:1 }
const b = { a:1, b:2 }
const d = diffstory.diff(a,b)

// d is { '+prop':{b:2}, '&prop':{a:1} }

Removals

const a = { k1:1, k2:2 }
const b = { k1:1 }
const d = diffstory.diff(a,b)

// d is { '-prop':{k2:2}, '&prop':{k1:1} }

Updates

const a = { k1:1 }
const b = { k1:2 }
const d = diffstory.diff(a,b)

// d is { '!prop': { k1: { '-val':1, '+val':2 } } }

Diffing Arrays

Additions

const a = [1]
const b = [1,2]
const d = diffstory.diff(a,b)

// d is [ {'&item':[1]}, {'+item':[2]} ]

Removals

const a = [1,2]
const b = [1]
const d = diffstory.diff(a,b)

// d is [ {'&item':[1]}, {'-item':[2]} ]

Updates

const a = [{k:1}]
const b = [{k:2}]
const d = diffstory.diff(a,b)

// d is [ {'!item': {'!prop': {k: { '-val':1, '+val':2 } } } } ]

Diffing Arrays of Objects

If you pass in the option { sameId : (a,b) => {..} }, then when diffing arrays of objects, the algorithm will be able to recognize when an object has been changed. Otherwise, it removes that object and adds another.

Diffing Strings

For strings, a "longest common subsequence" algorithm is used to produce diffs.

const d = diffstory.diff('ab','bc')

// d is [
//   { "-str": "a" },
//   { "&str": "b" },
//   { "+str": "c" },
// ]

If you want to diff on words instead of characters, pass in the following option

const d = diffstory.diff(
  'one two',
  'two three',
  { words: true },
)

// d is [
//   { "-str": "one " },
//   { "&str": "two" },
//   { "+str": " three" },
// ]

Diffing Strings, Compactly

There's also a more compact form of diff for strings.

const d = diffstory.diffStringsCompact('ab','bc')

// d is '-"a"&"b"+"c"' (- means `remove`, & means `keep`, + means `add`)

Applying Diffs

Forward

const a = "ab"
const b = "a"
const d = diffstory.diff(a,b)
bb = diffstory.forward(a,d)

// b === bb

Backward

const a = "ab"
const b = "a"
const d = diffstory.diff(a,b)
a = diffstory.backward(a,d)

// a === aa

Verifying Diffs

To ensure integrity, you can verify that a diff works forwards and backwards.

const a = "ab"
const b = "a"
const d = diffstory.diff(a,b)
const ok = diffstory.verify(a,b,d)

// ok === true

Future work

  • Compact diffs of multiple-member reorderings when the members are objects or arrays (currently handles just single-member reorderings)
  • Compact handling of array member duplication when the members are objects or arrays (currently treated as additions)

About

Javascript lib for producing diffs of data structures and time-traveling with them

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published