Metadata-Version: 2.1
Name: data-diff
Version: 0.0.3
Summary: A cross-database, efficient diff between mostly-similar database tables
Home-page: https://github.com/datafold/data-diff
License: MIT
Author: Erez Shinnan
Author-email: erezshin@gmail.com
Requires-Python: >=3.7,<4.0
Classifier: Development Status :: 2 - Pre-Alpha
Classifier: Environment :: Console
Classifier: Intended Audience :: Developers
Classifier: Intended Audience :: Information Technology
Classifier: Intended Audience :: System Administrators
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.7
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Topic :: Database :: Database Engines/Servers
Classifier: Typing :: Typed
Provides-Extra: mysql
Provides-Extra: pgsql
Provides-Extra: preql
Provides-Extra: snowflake
Requires-Dist: click (>=8.1,<9.0)
Requires-Dist: dsnparse
Requires-Dist: mysql-connector-python; extra == "mysql"
Requires-Dist: preql (>=0.2.11,<0.3.0); extra == "preql"
Requires-Dist: psycopg2; extra == "pgsql"
Requires-Dist: pyparsing
Requires-Dist: runtype (>=0.2.4,<0.3.0)
Requires-Dist: snowflake-connector-python; extra == "snowflake"
Project-URL: Repository, https://github.com/datafold/data-diff
Description-Content-Type: text/markdown

# Data Diff

A cross-database, efficient diff between mostly-similar database tables.

Use cases:

- Quickly validate that a table was copied correctly

- Find changes between two versions of the same table

We currently support the following databases:

- PostgreSQL

- MySQL

- Oracle

- Snowflake

- BigQuery

- Redshift


# How does it work?

Data Diff finds the differences between two tables by utilizing checksum calculations and logarithmic search.

Instead of comparing the entire table, it compares the tuple (primary_key, version_column), where the primary key is a unique identifier of the rows, and the version_column updates each time the row changes to a new value, that is unique to that update. Usually the versioning column would be a timestamp like `updated_at`, that would auto-update by the database. But it could also be an auto-counting integer, and so on.

Data Diff runs a checksum on these columns using MD5. If the checksums are not the same, we know the tables are different. We then split each table into "n" different segments of similar size (determined by the bisection factor), and repeat the comparison for each matching pair of segments. When segments are below a certain size (bisection threshold), we instead download the segments to the client, and diff them locally.

Data Diff splits the segments using "checkpoints", to ensure that inserted or deleted rows don't affect the quality of the diff.

This process is incremental, so differences are printed to stdout as they are found. Users can ensure Data Diff quits after finding some number of differences, either by providing the `--limit` option, or by closing the pipe (for example, by piping to `head`).

The algorithm goes like this:

0. Table segments `A` and `B` are set to the two tables for comparison.

1. Calculate the checksums on `A` and `B` using MD5.

    1. If they are the same, the tables are considered equal. Stop.

    2. If their size is below the threshold, diff them locallly and print the results.

    3. Else:  (they are different and above the threshold)

        1. Select `n-1` rows (checkpoints) in table `A`, splitting it into `n` segments of similar size.

        2. Filter out checkpoints that don't exist in table `B`.

        3. Split both `A` and `B` into `m <= n` segments according to the mutual checkpoints. `m` must be at least 2.

        4. For each pair of segments `Ai` and `Bi` (where `0 <= i <= m`), recurse into step 1.

## Example

The following printout shows the diff of two tables, Original and Original_1diff, with 25 million rows each, and just 1 different row between them.

We ran it with a very low bisection factor, and with the verbose flag, to demonstrate how it works.

Note: It's usually much faster to use high bisection factors, especially when there are very few changes, like in this example.

```python
$ data_diff postgres:/// Original  postgres:/// Original_1diff  -v --bisection-factor=4
[16:55:19] INFO - Diffing tables of size 25000095 and 25000095 | segments: 4, bisection threshold: 1048576.
[16:55:36] INFO - Diffing segment 0/4 of size 8333364 and 8333364
[16:55:45] INFO - . Diffing segment 0/4 of size 2777787 and 2777787
[16:55:52] INFO - . . Diffing segment 0/4 of size 925928 and 925928
[16:55:54] INFO - . . . Diff found 2 different rows.
+ (20000, 942013020)
- (20000, 942013021)
[16:55:54] INFO - . . Diffing segment 1/4 of size 925929 and 925929
[16:55:55] INFO - . . Diffing segment 2/4 of size 925929 and 925929
[16:55:55] INFO - . . Diffing segment 3/4 of size 1 and 1
[16:55:56] INFO - . Diffing segment 1/4 of size 2777788 and 2777788
[16:55:58] INFO - . Diffing segment 2/4 of size 2777788 and 2777788
[16:55:59] INFO - . Diffing segment 3/4 of size 1 and 1
[16:56:00] INFO - Diffing segment 1/4 of size 8333365 and 8333365
[16:56:06] INFO - Diffing segment 2/4 of size 8333365 and 8333365
[16:56:11] INFO - Diffing segment 3/4 of size 1 and 1
[16:56:11] INFO - Duration: 53.51 seconds.
```


# How to install

Requires Python 3.7+ with pip.

    poetry build --format wheel
    pip install "dist/data_diff-0.0.2-py3-none-any.whl[mysql,pgsql]"

# How to use

Usage: `data_diff DB1_URI TABLE1_NAME DB2_URI TABLE2_NAME [OPTIONS]`

Options:

  - `--help` - Show help message and exit.
  - `-k` or `--key_column` - Name of the primary key column
  - `-c` or `--columns` - List of names of extra columns to compare
  - `-l` or `--limit` - Maximum number of differences to find (limits maximum bandwidth and runtime)
  - `-s` or `--stats` - Print stats instead of a detailed diff
  - `-d` or `--debug` - Print debug info
  - `-v` or `--verbose` - Print extra info
  - `--bisection-factor` - Segments per iteration. When set to 2, it performs binary search.
  - `--bisection-threshold` - Minimal bisection threshold. i.e. maximum size of pages to diff locally.

## Tips for performance

It's highly recommended that all involved columns are indexed.

# License

TBD
