@labig/rfc6902 v3.0.4
rfc6902
Complete implementation of RFC6902 "JavaScript Object Notation (JSON) Patch"
(including RFC6901 "JavaScript Object Notation (JSON) Pointer"),
for creating and consuming application/json-patch+json documents.
Also offers "diff" functionality without using Object.observe.
Quickstart
Install locally:
npm install --save rfc6902Import in your script:
var rfc6902 = require('rfc6902')Calculate diff between two objects:
rfc6902.createPatch({first: 'Chris'}, {first: 'Chris', last: 'Brown'})
//⇒ [ { op: 'add', path: '/last', value: 'Brown' } ]Apply a patch to some object:
var users = [{first: 'Chris', last: 'Brown', age: 20}]
rfc6902.applyPatch(users, [
{op: 'replace', path: '/0/age', value: 21},
{op: 'add', path: '/-', value: {first: 'Raphael', age: 37}},
])The applyPatch function returns [null, null],
indicating there were two patches, both applied successfully.
The users variable is modified in place; evaluate it to examine the end result:
users
//⇒ [ { first: 'Chris', last: 'Brown', age: 21 },
// { first: 'Raphael', age: 37 } ]API
In ES6 syntax:
import {applyPatch, createPatch} from 'rfc6902'Using TypeScript annotations for clarity:
applyPatch(object: any, patch: Operation[]): Array<Error | null>
The operations in patch are applied to object in-place.
Returns a list of results as long as the given patch.
If all operations were successful, each item in the returned list will be null.
If any of them failed, the corresponding item in the returned list will be an Error instance
with descriptive .name and .message properties.
createPatch(input: any, output: any, diff?: VoidableDiff): Operation[]
Returns a list of operations (a JSON Patch) of the required operations to make input equal to output.
In most cases, there is more than one way to transform an object into another.
This method is more efficient than wholesale replacement,
but does not always provide the optimal list of patches.
It uses a simple Levenshtein-type implementation with Arrays,
but it doesn't try for anything much smarter than that,
so it's limited to remove, add, and replace operations.
The optional diff argument allows the user to specify a partial function
that's called before the built-in diffAny function.
For example, to avoid recursing into instances of a custom class, say, MyObject:
function myDiff(input: any, output: any, ptr: Pointer) {
if ((input instanceof MyObject || output instanceof MyObject) && input != output) {
return [{op: 'replace', path: ptr.toString(), value: output}]
}
}
const my_patch = createPatch(input, output, myDiff)This will short-circuit on encountering an instance of MyObject, but otherwise recurse as usual.
Operation
interface Operation {
op: 'add' | 'remove' | 'replace' | 'move' | 'copy' | 'test'
from?: string
path?: string
value?: string
}Different operations use different combinations of from / value;
see JSON Patch (RFC6902) below.
Demo
Simple web app using the browser-compiled version of the code.
Determinism
If you've ever implemented Levenshtein's algorithm,
or played tricks with git rebase to get a reasonable sequence of commits,
you'll realize that computing diffs is rarely deterministic.
E.g., to transform the string ab → bc, you could:
1. Delete a (⇒ b)
2. and then append c (⇒ bc)
Or...
1. Replace b with c (⇒ ac)
2. and then replace a with b (⇒ bc)
Both consist of two operations, so either one is a valid solution.
Applying json-patch documents is much easier than generating them,
which might explain why, when I started this project,
there were more than five patch-applying RFC6902 implementations in NPM,
but none for generating a patch from two distinct objects.
(There was one that used Object.observe(), which only works when you're the one making the changes,
and only as long as Object.observe() hasn't been deprecated, which it has.)
So when comparing your data objects, you'll want to ensure that the patches it generates meet your needs. The algorithm used by this library is not optimal, but it's more efficient than the strategy of wholesale replacing everything that's not an exact match.
Of course, this only applies to generating the patches. Applying them is deterministic and unambiguously specified by RFC6902.
Tutorial
JSON Pointer (RFC6901)
The RFC is a quick and easy read, but here's the gist:
- JSON Pointer is a system for pointing to some fragment of a JSON document.
- A pointer is a string that is composed of zero or more /reference-token parts.
- When there are zero (the empty string), the pointer indicates the entire JSON document.
- Otherwise, the parts are read from left to right, each one selecting part of the current document, and presenting only that fragment of the document to the next part.
- The reference-token bits are usually Object keys, but may also be (decimal) numerals, to indicate array indices.
E.g., consider the NPM registry:
{
"_updated": 1417985649051,
"flickr-with-uploads": {
"name": "flickr-with-uploads",
"description": "Flickr API with OAuth 1.0A and uploads",
"repository": {
"type": "git",
"url": "git://github.com/chbrown/flickr-with-uploads.git"
},
"homepage": "https://github.com/chbrown/flickr-with-uploads",
"keywords": [
"flickr",
"api",
"backup"
],
...
},
...
}/_updated: this selects the value of that key, which is just a number:1417985649051/flickr-with-uploads: This selects the entire object:{ "name": "flickr-with-uploads", "description": "Flickr API with OAuth 1.0A and uploads", "repository": { "type": "git", "url": "git://github.com/chbrown/flickr-with-uploads.git" }, "homepage": "https://github.com/chbrown/flickr-with-uploads", "keywords": [ "flickr", "api", "backup" ], ... }/flickr-with-uploads/name: this effectively applies the/namepointer to the result of the previous item, which selects the string,"flickr-with-uploads"./flickr-with-uploads/keywords/1: Array indices start at 0, so this selects the second item from thekeywordsarray, namely,"api".
Rules:
- A pointer, if it is not empty, must always start with a slash; otherwise, it is an "Invalid pointer syntax" error.
- If a key within the JSON document contains a forward slash character
(which is totally valid JSON, but not very nice),
the
/in the desired key should be replaced by the escape sequence,~1. - If a key within the JSON document contains a tilde (again valid JSON, but not very common),
the
~should be replaced by the other escape sequence,~0. This allows keys containing the literal string~1(which is especially cruel) to be referenced by a JSON pointer (e.g.,/~01should returntruewhen applied to the object{"~1":true}). - All double quotation marks, reverse slashes, and control characters must escaped, since a JSON Pointer is a JSON string.
- A pointer that refers to a non-existent value counts as an error, too. But not necessarily as fatal as a syntax error.
Example
This project implements JSON Pointer functionality in rfc6902/pointer; e.g.:
const {Pointer} = require('rfc6902/pointer')
const repository = {
contributors: ['chbrown', 'diachedelic', 'nathanrobinson', 'kbiedrzycki', 'stefanmaric']
}
const pointer = Pointer.fromJSON('/contributors/0')
//⇒ Pointer { tokens: [ '', 'contributors', '0' ] }
pointer.get(repository)
//⇒ 'chbrown'JSON Patch (RFC6902)
The RFC is only 18 pages long, but here are the basics:
A JSON Patch document is a JSON document such that:
- The MIME Type is
application/json-patch+json - The file extension is
.json-patch - It is an array of patch objects, potentially empty.
- Each patch object has a key,
op, with one of the following six values, and an operator-specific set of other keys.add: Insert the givenvalueatpath. Or replace it, if it already exists. If the parent of the intended target does not exist, produce an error. If the final reference-token ofpathis "-", and the parent is an array, appendvalueto it.path: JSON Pointervalue: JSON object
remove: Remove the value atpath. Produces an error if it does not exist. Ifpathrefers to an element within an array, splice it out so that subsequent elements fill in the gap (decrementing the length of the array).path: JSON Pointer
replace: Replace the current value atpathwithvalue. It's exactly the same as performing aremoveoperation and then anaddoperation on the same path, since there must be a pre-existing value.path: JSON Pointervalue: JSON object
move: Remove the value atfrom, and setpathto that value. There must be a value atfrom, but not necessarily atpath; it's the same as performing aremoveoperation, and then anaddoperation, but on different paths.from: JSON Pointerpath: JSON Pointer
copy: Get the value atfromand setpathto that value. Same asmove, but doesn't remove the original value.from: JSON Pointerpath: JSON Pointer
test: Check that the value atpathis equal tovalue. If it is not, the entire patch is considered to be a failure.path: JSON Pointervalue: JSON object
License
Copyright 2014-2019 Christopher Brown. MIT Licensed.
6 years ago