Skip to content

Open-Tech-Foundation/obj-diff

Folders and files

NameName
Last commit message
Last commit date

Latest commit

ย 

History

75 Commits
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 

@opentf/obj-diff

The Fast, Accurate, and Modern JavaScript Objects Diffing & Patching Library.

CI Status JSR Score

Part of the Open Tech Foundation ecosystem.

Demo image

Live Demo | Report Bug | Standard Library


๐Ÿš€ Features

  • ๐Ÿ” Deep Objects Diffing: Detects changes at any depth.
  • ๐Ÿฉน Efficient Patching: Apply diffs to recreate target objects.
  • ๐Ÿ› ๏ธ Extensible: Support for custom object types via diffWith().
  • ๐Ÿ“ฆ Modern Ecosystem: Built for Bun, Node.js, Deno, and Browser.
  • ๐ŸŸฆ TypeScript Native: Full type safety and autocompletion.
  • โšก High Performance: Optimized for speed and minimal memory footprint.

๐Ÿ“ฆ Installation

Install @opentf/obj-diff using your preferred package manager:

# Bun
bun add @opentf/obj-diff

# pnpm
pnpm add @opentf/obj-diff

# npm
npm install @opentf/obj-diff

# Deno
deno add @opentf/obj-diff

๐Ÿ›  Supported Types

The library natively supports the following types:

  • Primitives: Undefined, Null, Number, String, Boolean, BigInt.
  • Built-in Objects: Plain Objects {}, Array, Date, Map, Set.

๐Ÿ“– Usage

diff(obj1, obj2)

Performs a deep comparison between two objects.

import { diff } from '@opentf/obj-diff';

const result = diff(obj1, obj2);

DiffResult Structure

type DiffResult = {
  type: 0 | 1 | 2;              // 0: Deleted, 1: Created, 2: Updated
  path: Array<string | number>; // The path to the property
  value?: unknown;              // The value (for Created/Updated)
};

patch(obj, patches)

Applies an array of diff results to an object.

import { patch } from "@opentf/obj-diff";

const updatedObj = patch(originalObj, diffResults);

๐Ÿ’ก Examples

1. Basic Objects

const a = { a: 1, b: 2 };
const b = { a: 2, c: 3 };

diff(a, b);
/*
[
  { type: 2, path: ["a"], value: 2 },
  { type: 0, path: ["b"] },
  { type: 1, path: ["c"], value: 3 }
]
*/

2. Nested Structures

const a = { foo: { bar: [1, 2] } };
const b = { foo: { bar: [1] } };

const d = diff(a, b);
const res = patch(a, d); // res is deep equal to b

3. ES6 Map & Set Support

Natively diff and patch modern collections.

const a = new Set([1, 2]);
const b = new Set([2, 3]);

diff(a, b);
/*
[
  { type: 0, path: [0], value: 1 },
  { type: 1, path: [1], value: 3 }
]
*/

4. Circular Reference Safety

Safe comparison of recursive objects without infinite loops.

const a = { id: 1 };
a.self = a;
const b = { id: 2 };
b.self = b;

diff(a, b); 
// Output: [{ type: 2, path: ["id"], value: 2 }]

5. Custom Types via diffWith()

Extend the diffing logic for specialized types like MongoDB ObjectId.

import { diffWith } from "@opentf/obj-diff";
import { ObjectId } from "bson";

const result = diffWith(record1, record2, (a, b) => {
  if (a instanceof ObjectId && b instanceof ObjectId) {
    return a.toString() !== b.toString();
  }
});

โš ๏ธ Caveats

Internal Object Sharing (Aliasing)

For maximum performance, @opentf/obj-diff preserves internal object identity (sharing) during the patch() operation.

If your original object contains multiple paths pointing to the same object instance, patching one of those paths will affect all its aliases.

const shared = { x: 1 };
const a = { first: shared, second: shared };
const b = { first: { x: 1 }, second: { x: 2 } };

const d = diff(a, b);
const res = patch(a, d);

// res.first.x will be 2 because it shares the same instance as res.second
console.log(res.first.x); // 2

Tip

If you require independent branches after patching, ensure your input objects do not share internal references that are expected to diverge.


๐Ÿ“Š Benchmark

We prioritize performance without sacrificing accuracy.

Library Ops/sec Average Time Notes
@opentf/obj-diff 246,154 ~4.0ฮผs Fastest; Full Diff + Patch support.
microdiff 158,745 ~6.3ฮผs Very fast; No patching support.
jsondiffpatch 157,453 ~6.3ฮผs Rich features (LCS, RFC6902); Slower.
deep-object-diff 151,559 ~6.6ฮผs Fast; Basic diffing only.
deep-diff 111,615 ~9.0ฮผs Medium; Classic library.
recursive-diff 79,628 ~12.5ฮผs Slower; Good for complex recursion.
just-diff 66,477 ~15.0ฮผs Slowest in this test.

Running Benchmarks Locally

bun run build
bun benchmark.js

โ“ FAQs

1. Why is JSON Patch (RFC 6902) not supported?

The JSON Patch protocol is quite heavy and complex. We've optimized @opentf/obj-diff for performance and simplicity, which covers the vast majority of real-world use cases.

2. What does an empty path path: [] mean?

An empty path denotes the Root of the object. It typically means the entire source was replaced by the target value (e.g., comparing an object to null).


๐Ÿ“– Articles

Explore the philosophy behind our standard library:


๐Ÿ“„ License

This project is licensed under the MIT License.

About

๐Ÿš€ The Fast, Accurate, JavaScript Objects Diffing & Patching Library.

Topics

Resources

License

Code of conduct

Contributing

Stars

Watchers

Forks

Sponsor this project

 

Packages

 
 
 

Contributors