SkillAgentSearch skills...

Patchdiff

Patchdiff 🔍 Bidirectional patch diffs

Install / Use

/learn @fork-tongue/Patchdiff
About this skill

Quality Score

0/100

Supported Platforms

Universal

README

PyPI version CI status

Patchdiff 🔍

Based on rfc6902 this library provides a simple API to generate bi-directional diffs between composite python datastructures composed out of lists, sets, tuples and dicts. The diffs are jsonpatch compliant, and can optionally be serialized to json format. Patchdiff can also be used to apply lists of patches to objects, both in-place or on a deepcopy of the input.

Install

pip install patchdiff

Quick-start

from patchdiff import apply, diff, iapply, to_json

input = {"a": [5, 7, 9, {"a", "b", "c"}], "b": 6}
output = {"a": [5, 2, 9, {"b", "c"}], "b": 6, "c": 7}

ops, reverse_ops = diff(input, output)

assert apply(input, ops) == output
assert apply(output, reverse_ops) == input

iapply(input, ops)  # apply in-place
assert input == output

print(to_json(ops, indent=4))
# [
#     {
#         "op": "add",
#         "path": "/c",
#         "value": 7
#     },
#     {
#         "op": "replace",
#         "path": "/a/1",
#         "value": 2
#     },
#     {
#         "op": "remove",
#         "path": "/a/3/a"
#     }
# ]

Proxy-based patch generation

For better performance, produce() can be used which generates patches by tracking mutations on a proxy object (inspired by Immer):

from patchdiff import produce

base = {"count": 0, "items": [1, 2, 3]}

def recipe(draft):
    """Mutate the draft object - changes are tracked automatically."""
    draft["count"] = 5
    draft["items"].append(4)
    draft["new_field"] = "hello"

result, patches, reverse_patches = produce(base, recipe)

# base is unchanged (immutable by default)
assert base == {"count": 0, "items": [1, 2, 3]}

# result contains the changes
assert result == {"count": 5, "items": [1, 2, 3, 4], "new_field": "hello"}

# patches describe what changed
print(patches)
# [
#     {"op": "replace", "path": "/count", "value": 5},
#     {"op": "add", "path": "/items/-", "value": 4},
#     {"op": "add", "path": "/new_field", "value": "hello"}
# ]

When immutability is not needed, it is possible to apply the ops directly, improving performance even further by not having to make a deepcopy of the given state.

from observ import reactive
from patchdiff import produce

state = reactive({"count": 0})

# Mutate in place and get patches for undo/redo
result, patches, reverse = produce(
    state, 
    lambda draft: draft.update({"count": 5}), 
    in_place=True,
)

assert result is state  # Same object
assert state["count"] == 5  # State was mutated, watchers triggered

Related Skills

View on GitHub
GitHub Stars7
CategoryDevelopment
Updated11d ago
Forks0

Languages

Python

Security Score

70/100

Audited on Mar 17, 2026

No findings