PDQ hash and URL duplicate detector. Developed by Sam Sweere from BigData Repulic as part of their Social Good Initiaive.
Project description
Duplicate Detector - for CIR
As part of BigData Republic Social Good Initiative, Sam Sweere developed this package to help the CIR team to detect & mark (possible) duplicate entries. There are two kinds of duplicates that the package can detect:
- PDQ hash fuzzy duplicates: These are entries with similar PDQ hashes, and are likely to be similar images.
- URL duplicates: These are entries that have the same URL. The url is parsed such that the parameters are removed and the url is normalized.
PDQ Duplicate Detection
The PDQ algorithm was developed and open-sourced by Facebook (now Meta) in 2019. It is a perceptual hash function that can be used to identify similar images. For example, if an image is cropped, resized, or slightly altered, the PDQ hash will still be similar to the original image.
For open source intelligence use cases, the PDQ hash can be used to identify similar images across different sources. For example, if an image is shared on social media and in a news article, but the news article puts their watermark on the image, the PDQ hash can still be used to identify the image as the same.
While working at BigData Republic, Emiel de Heij has implemented the PDQ hash in the Bellingcat auto-archiver. From version v0.5.26 onwards, the auto-archiver has the ability to automatically generate the PDQ hashes for images and videos.
Usage
The detect_duplicates
function finds url and/or pdq hash duplicates in a DataFrame.
It takes the following parameters:
Input:
-
DataFrame (
pd.DataFrame
):- Columns:
index
(string): A unique identifier.url
(string, optional): The original URL of the source.pdq_hash
(list of strings, optional): A list of perceptual hashes.
- Columns:
-
Indices to Check (
indices_to_check
):- Type: List of strings or
None
(optional). - Description: Specific
index
values to check for duplicates. IfNone
, all rows are checked. Duplicate annotations are always bi-directional.
- Type: List of strings or
-
PDQ Hash Similarity Threshold (
pdq_hash_similarity_threshold
):- Type: Float (between 0 and 1, optional, default: 0.9). #TODO: update this
- Description: The threshold for the Hamming distance to determine hash similarity.
-
PDQ Duplicate Detection Method (
pdq_duplicate_detection_method
):- Type: String (optional, default: "pandas").
- Description: The method to use for detecting PDQ hash duplicates. Options are "pandas" and "bk-tree".
Output:
- DataFrame (
pd.DataFrame
):- Columns:
index
(string): A unique identifier.url_duplicates
(list of strings orNone
): Indices with duplicate URLs.pdq_hash_duplicates
(list of strings orNone
): Indices with perceptual hashes similar within the threshold.pdq_hash_similarity
(list of strings orNone
): Hash similarity scores for perceptual hashes within the threshold.
- Columns:
Post-processing
When indices_to_check
is provided, only the rows with these indices are checked. However, duplicates can be identified with indices not in this list (bi-directional). Consider this when updating the source. Note that only rows with any duplicate found will be returned.
Example Usage
from cir_duplicate_detector import detect_duplicates
duplicates_df = detect_duplicates(df, indices_to_check=None, pdq_hash_similarity_threshold=0.8)
PDQ Hash Similarity Detection Approach
This package offers two strategies for identifying duplicate hashes, leveraging multi-threading to enhance performance on systems with multiple cores:
pandas
: Utilizing the.apply
function, this strategy performs a straightforward one-to-one comparison of hashes by calculating the hamming distance with therapidfuzz
library. It proves to be the quickest and most efficient method for smaller datasets (i.e., when theindices_to_check
parameter is minimal).bk-tree
: This approach employs a BK-tree to store and search for hashes. While it is less efficient than the pandas approach for lower similarity thresholds, it is vastly more efficient for higher thresholds. When processing larger datasets, the bk-tree method's performance is comparable to that of the pandas strategy.
Performance Comparison
This section outlines the performance of the two methods, focusing on execution speed, benchmarked on a dataset of approximately 300,000 hashes on a system equipped with 20 CPU cores. Both methods employ multi-threading to optimize utilization of available computational resources.
Runtime vs Similarity Threshold
The comparison indicates that, for a similarity threshold up to 0.91, the BK-tree method operates at a slower pace compared to the pandas method. Beyond this threshold, however, the BK-tree method demonstrates a notable increase in speed over the pandas method. This performance inversion is due to the BK-tree's complexity in managing hashes with a greater number of allowable bits, requiring extensive traversal to locate hashes matching the query. In contrast, the pandas method does not face such traversal overhead, maintaining a consistent speed advantage under these conditions.
Given that the optimal threshold for identifying fuzzy duplicates tends to be more towards 0.8, the pandas method is recommended for most scenarios and thus is the default choice.
Runtime vs Number of Hashes Checked
Here, the BK-tree method's performance is observed to lag behind that of the pandas method at a similarity threshold of 0.8. This performance gap widens as the number of hashes to be checked decreases. This outcome stems from the necessity for the BK-tree method to construct the tree structure before initiating the search process. The pandas method is more efficient, especially for smaller datasets, because it doesn't require this initial setup.
Development Installation
Make sure to have Python 3.11 installed and create a virtual environment.
Install the requirements by using:
pip install -r requirements.txt
Detailed Development Installation
The following steps are for a detailed installation of the development environment. Note that for every step there are multiple ways to i.e. install python, create an environment or install dependencies. The following steps are just one way to do it.
-
Install
Pyenv
: https://github.com/pyenv/pyenv#installation -
Install
python 3.11.8
:pyenv install 3.11.8
-
Install pyenv-virtualenv: https://github.com/pyenv/pyenv-virtualenv
-
Create a virtual environment:
pyenv virtualenv 3.11.8 cir_duplicate_detector
-
Enable and use the virtual environment:
pyenv local cir_duplicate_detector pyenv activate cir_duplicate_detector
-
Install poetry:
pip install poetry
-
Install the dependencies:
poetry install
Contributors
- Sam Sweere: Code implementation, testing, documentation, and more.
- Emiel de Heij: Project initiation, PDQ hash implementation in the Bellingcat auto-archiver.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file cir_duplicate_detector-0.1.3.tar.gz
.
File metadata
- Download URL: cir_duplicate_detector-0.1.3.tar.gz
- Upload date:
- Size: 17.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.8.2 CPython/3.9.12 Linux/6.5.0-18-generic
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 9e9f610934afccd5724df1817dfca0efeb8807cd1f4751920ad21d4ba7fa7125 |
|
MD5 | 716ff7143876317aad24d5f4f8fe7a91 |
|
BLAKE2b-256 | c7855ac344b17281138c6e05d92a71173b0f130d46da354a0f1dea3e5fd02488 |
File details
Details for the file cir_duplicate_detector-0.1.3-py3-none-any.whl
.
File metadata
- Download URL: cir_duplicate_detector-0.1.3-py3-none-any.whl
- Upload date:
- Size: 17.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.8.2 CPython/3.9.12 Linux/6.5.0-18-generic
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 447b0dfb871fe3720ff635b643e5171d451e533566b5b8c6264c2631b03ff36c |
|
MD5 | 4cf6bba8516adfc8db4b04729d3fbe22 |
|
BLAKE2b-256 | 4c4ac1f8cd3a6c28cc671b23bbf64f5a422f084f360492a157cf1f911b50bc17 |