a structural diff that understands syntax 🟥🟩
Go to file
2024-11-15 23:17:27 -08:00
.github Bump MSRV to 1.66 2024-10-15 23:15:07 -07:00
demo_files Find and replace all instances of "the all the" with "all the" 2024-08-26 08:17:51 -07:00
homepage Add logos to homepage 2024-04-07 16:23:03 -07:00
img Update screenshot and ensure sample files match 2023-01-22 20:19:36 -08:00
manual docs(contributing): Add mdbook install note about toolchain 2024-10-30 07:45:02 -07:00
sample_files Improve syntax highlighting on tree-sitter lists that are just keywords 2024-11-15 00:03:30 -05:00
src Ensure JSON output is consistently ordered 2024-11-15 23:17:27 -08:00
tests Fix crash when the last hunk includes the trailing newline at EOF 2024-07-26 08:54:20 -07:00
translation/zh-CN Bump GitHub action workflows to their latest versions 2024-09-11 21:22:59 -07:00
vendored_parsers Merge commit 'a13f2d1ee9609cc5c4c8ffce9640c353b77a24d8' 2024-07-23 12:49:30 +02:00
.codecov.yml Disable codecov comments on PRs 2022-04-22 00:53:48 -07:00
.gitattributes chore: fix crlf line endings 2024-04-28 04:25:52 -04:00
.gitignore chore: generate 2024-04-14 16:03:44 -04:00
build.rs Merge commit 'b5dbafd164af7113e208de9b36068046f5ce8678' into f_sharp 2024-05-14 10:08:50 -07:00
Cargo.lock Roll version 2024-10-24 08:17:18 -07:00
Cargo.toml Roll version 2024-10-24 08:17:18 -07:00
CHANGELOG.md Improve handling of named pipe arguments 2024-11-15 23:11:25 -08:00
CITATION.cff Add citation information 2021-11-23 09:52:39 -08:00
difft.1 Expand man page and add to changelog 2024-04-28 15:07:23 -07:00
difft.1.md Expand man page and add to changelog 2024-04-28 15:07:23 -07:00
justfile Write a basic man page 2024-04-27 16:21:17 -07:00
LICENSE Happy new year! 2024-01-04 17:37:46 -08:00
README.md Document the usage of bright colours and Solarized 2024-07-30 15:02:09 -07:00
rust-toolchain.toml Bump MSRV to 1.66 2024-10-15 23:15:07 -07:00

it's difftastic!
English manual Chinese manual crates.io codecov.io

Difftastic is a structural diff tool that compares files based on their syntax.

For installation instructions, see Installation in the manual.

Basic Example

Screenshot of difftastic and JS

In this JavaScript example, we can see:

(1) Difftastic understands nesting. It highlights the matching { and }, but understands that foo() hasn't changed despite the leading whitespace.

(2) Difftastic understands which lines should be aligned. It's aligned bar(1) on the left with bar(2) on the right, even though the textual content isn't identical.

(3) Difftastic understands that line-wrapping isn't meaningful. "eric" is now on a new line, but it hasn't changed.

One Minute Demo

asciicast

This one minute screencast demonstrates difftastic usage with both standalone files and git.

Languages

Difftastic supports over 30 programming languages, see the manual for the full list.

If a file has an unrecognised extension, difftastic uses a textual diff with word highlighting.

Known Issues

Performance. Difftastic scales relatively poorly on files with a large number of changes, and can use a lot of memory.

Display. Difftastic has a side-by-side display which usually works well, but can be confusing.

Robustness. Difftastic regularly has releases that fix crashes.

Non-goals

Patching. Difftastic output is intended for human consumption, and it does not generate patches that you can apply later. Use diff if you need a patch.

(Patch files are also line-oriented, which is too limited for difftastic. Difftastic might find additions and removals on the same line, and it tracks the relationship between line numbers in the old and new file.)

Merging. AST merging is a hard problem that difftastic does not address.

FAQ

Isn't this basically --word-diff --ignore-all-space?

Word diffing can't do this.

Difftastic parses your code. It understands when whitespace matters, such as inside string literals or languages like Python. It understands that x-1 is three tokens in JS but one token in Lisp.

Can I use difftastic with git?

You can! The difftastic manual includes instructions for git usage. You can also use it with mercurial.

If you're a magit user, check out this blog post showing one way to use difftastic with magit.

Does difftastic integrate with my favourite tool?

Probably not. Difftastic is young. Consider writing a plugin for your favourite tool, and I will link it in the README!

Can difftastic help me with merge conflicts?

Yes! As of version 0.50, difftastic understands merge conflict markers (i.e. <<<<<<<, ======= and >>>>>>>).

Pass your file with conflicts as a single argument to difftastic. Difftastic will construct the two conflicting files and diff those.

$ difft file_with_conflicts.js

Can difftastic do merges?

No. AST merging is a hard problem that difftastic does not address.

AST diffing is a lossy process from the perspective of a text diff. Difftastic will ignore whitespace that isn't syntactically significant, but merging requires tracking whitespace.

Can difftastic ignore reordering?

No. Difftastic always considers order to be important, so diffing e.g. set(1, 2) and set(2, 1) will show changes.

If you're diffing JSON, consider sorting the keys before passing them to difftastic.

$ difft <(jq --sort-keys < file_1.json) <(jq --sort-keys < file_2.json)

See also Tricky Cases: Unordered Data Types in the manual.

Can I use difftastic to check for syntactic changes without diffing?

Yes. Difftastic can check if the two files have the same AST, without calculating a diff. This is much faster than normal diffing, and useful for building tools that check for changes.

For example:

$ difft --check-only --exit-code before.js after.js

This will set the exit code to 0 if there are no syntactic changes, or 1 if there are changes found.

Why aren't colours appearing in my terminal?

Difftastic uses ANSI bright colours by default, but some terminal themes show bright colours as grey. Solarized is a popular theme that does this.

If you're a Solarized user, use export DFT_BACKGROUND=light to disable bright colours, or try a different terminal colour scheme.

How does it work?

Difftastic treats structural diffing as a graph problem, and uses Dijkstra's algorithm.

My blog post describes the design, and there is also an internals section in the manual.

Translation

License

Difftastic is open source under the MIT license, see LICENSE for more details.

This repository also includes tree-sitter parsers by other authors in the vendored_parsers/ directory. These are a mix of the MIT license and the Apache license. See vendored_parsers/*/LICENSE for more details.

Files in sample_files/ are also under the MIT license unless stated otherwise in their header.