Metadata-Version: 2.4
Name: philiprehberger-json-diff
Version: 0.5.0
Summary: Readable JSON comparison with colorized terminal output.
Project-URL: Homepage, https://github.com/philiprehberger/py-json-diff#readme
Project-URL: Repository, https://github.com/philiprehberger/py-json-diff
Project-URL: Issues, https://github.com/philiprehberger/py-json-diff/issues
Project-URL: Changelog, https://github.com/philiprehberger/py-json-diff/blob/main/CHANGELOG.md
Author: Philip Rehberger
License-Expression: MIT
License-File: LICENSE
Keywords: colorized,compare,debugging,diff,json
Classifier: Development Status :: 3 - Alpha
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Typing :: Typed
Requires-Python: >=3.10
Description-Content-Type: text/markdown

# philiprehberger-json-diff

[![Tests](https://github.com/philiprehberger/py-json-diff/actions/workflows/publish.yml/badge.svg)](https://github.com/philiprehberger/py-json-diff/actions/workflows/publish.yml)
[![PyPI version](https://img.shields.io/pypi/v/philiprehberger-json-diff.svg)](https://pypi.org/project/philiprehberger-json-diff/)
[![Last updated](https://img.shields.io/github/last-commit/philiprehberger/py-json-diff)](https://github.com/philiprehberger/py-json-diff/commits/main)

Readable JSON comparison with colorized terminal output.

## Installation

```bash
pip install philiprehberger-json-diff
```

## Usage

```python
from philiprehberger_json_diff import diff, format_diff, diff_summary

old = {"name": "Alice", "age": 30, "city": "NYC"}
new = {"name": "Alice", "age": 31, "country": "US"}

changes = diff(old, new)

# Pretty-print with colors
print(format_diff(changes))

# Get a summary
summary = diff_summary(changes)
# {"added": 1, "removed": 1, "modified": 1, "unchanged": 0}
```

### Nested Comparison

```python
from philiprehberger_json_diff import diff

old = {"user": {"name": "Alice", "settings": {"theme": "dark"}}}
new = {"user": {"name": "Alice", "settings": {"theme": "light"}}}

changes = diff(old, new)
# Reports: modified user.settings.theme: "dark" -> "light"
```

### Ignore Paths

```python
from philiprehberger_json_diff import diff

changes = diff(old, new, ignore={"user.settings.theme"})
```

### Wildcard Ignore Patterns

```python
from philiprehberger_json_diff import diff

old = {"a": {"metadata": {"ts": 1}}, "b": {"metadata": {"ts": 2}}}
new = {"a": {"metadata": {"ts": 99}}, "b": {"metadata": {"ts": 99}}}

# Ignore metadata fields at any depth
changes = diff(old, new, ignore={"*.metadata.*"})
```

### Filter by paths

```python
from philiprehberger_json_diff import diff_paths

old = {
    "users": {"alice": {"email": "a@old.com"}, "bob": {"email": "b@old.com"}},
    "config": {"timeout": 30, "retries": 3},
}
new = {
    "users": {"alice": {"email": "a@new.com"}, "bob": {"email": "b@new.com"}},
    "config": {"timeout": 60, "retries": 3},
}

# Watch only specific subtrees — wildcards work just like ignore patterns
changes = diff_paths(old, new, ["users.*.email", "config.timeout"])
# Reports: users.alice.email, users.bob.email, config.timeout
```

### Structural Diff Mode

```python
from philiprehberger_json_diff import diff

old = {"a": 1, "b": "hello", "c": 3}
new = {"a": "one", "b": "world", "d": 4}

result = diff(old, new, mode="structural")
result.key_additions   # [Change(path='d', ...)]
result.key_removals    # [Change(path='c', ...)]
result.value_changes   # [Change(path='b', ...)]
result.type_changes    # [Change(path='a', ...)]
```

### Apply Patch

```python
from philiprehberger_json_diff import diff, apply_patch

old = {"name": "Alice", "age": 30}
new = {"name": "Bob", "age": 31}

changes = diff(old, new)
result = apply_patch(old, changes)
# result == {"name": "Bob", "age": 31}
```

### JSON Patch (RFC 6902)

```python
from philiprehberger_json_diff import diff, to_json_patch

old = {"name": "Alice", "age": 30}
new = {"name": "Bob", "age": 31, "active": True}

patch = to_json_patch(diff(old, new))
# [
#   {"op": "replace", "path": "/age", "value": 31},
#   {"op": "replace", "path": "/name", "value": "Bob"},
#   {"op": "add", "path": "/active", "value": True},
# ]

# Round-trip: apply the patch to reconstruct `new` from `old`
from philiprehberger_json_diff import apply_json_patch

apply_json_patch(old, patch) == new  # True
```

### Array Diff Strategies

```python
from philiprehberger_json_diff import diff, ArrayStrategy

old = {"tags": ["python", "json", "diff"]}
new = {"tags": ["diff", "json", "python"]}

# Default: order-sensitive (reports changes at each index)
changes = diff(old, new)

# Order-insensitive: treats arrays as unordered collections
changes = diff(old, new, array_strategy=ArrayStrategy.ORDER_INSENSITIVE)
# No changes reported — same elements, different order
```

### HTML Output

```python
from philiprehberger_json_diff import diff, format_html

changes = diff(old, new)
html = format_html(changes)
# Returns an HTML <table> with rows classed "added", "removed", or "modified"
```

## API

| Function / Class | Description |
|------------------|-------------|
| `diff(a, b, ignore, mode, array_strategy)` | Compare two dicts/lists, returns list of `Change` objects or `StructuralDiff` |
| `diff_paths(a, b, paths, *, array_strategy)` | Compare two dicts/lists and return only changes matching the supplied glob-style path patterns |
| `format_diff(changes, color)` | Format changes as readable string with optional ANSI colors |
| `format_html(changes)` | Format changes as an HTML table for web UIs |
| `diff_summary(changes)` | Return dict with counts by change type |
| `to_json_patch(changes)` | Convert changes to RFC 6902 JSON Patch format |
| `apply_json_patch(target, ops)` | Apply RFC 6902 ops (`add`/`remove`/`replace`) to a target — inverse of `to_json_patch` |
| `apply_patch(target, changes)` | Apply a diff result as a patch to reconstruct the modified object |
| `Change` | Dataclass with `path`, `change_type`, `old_value`, `new_value` |
| `ChangeType` | Enum: `ADDED`, `REMOVED`, `MODIFIED`, `UNCHANGED` |
| `StructuralDiff` | Dataclass with `key_additions`, `key_removals`, `value_changes`, `type_changes` |
| `ArrayStrategy` | Enum: `ORDER_SENSITIVE`, `ORDER_INSENSITIVE` |

## Development

```bash
pip install -e .
python -m pytest tests/ -v
```

## Support

If you find this project useful:

⭐ [Star the repo](https://github.com/philiprehberger/py-json-diff)

🐛 [Report issues](https://github.com/philiprehberger/py-json-diff/issues?q=is%3Aissue+is%3Aopen+label%3Abug)

💡 [Suggest features](https://github.com/philiprehberger/py-json-diff/issues?q=is%3Aissue+is%3Aopen+label%3Aenhancement)

❤️ [Sponsor development](https://github.com/sponsors/philiprehberger)

🌐 [All Open Source Projects](https://philiprehberger.com/open-source-packages)

💻 [GitHub Profile](https://github.com/philiprehberger)

🔗 [LinkedIn Profile](https://www.linkedin.com/in/philiprehberger)

## License

[MIT](LICENSE)
