diffstory
v0.2.2
Published
Javascript lib for producing diffs of data structures and time-traveling with them
Downloads
9
Readme
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)