Cython bindings and Python interface to SWORD (Smith Waterman On Reduced Database), a heuristic method for fast database search.
Project description
🐍🗡️ PySWRD
Cython bindings and Python interface to SWORD (Smith Waterman On Reduced Database), a method for fast database search.
🗺️ Overview
Searching a sequence inside a database of target sequences involves aligning the sequence to all the targets to find the highest scoring ones, which has a high computational cost. Several methods have been proposed over the years that use a pre-filter to select. In BLAST[1], k-mers are extracted from the query, and only targets containing high-scoring k-mers, with respect to the scoring matrix, are actually aligned.
SWORD[2] proposes a pre-filter built on perfect hashing of short mismatching k-mers. The k-mers generated from the query sequence also include k-mers with mismatches to improve sensitivity. When a k-mer is found in a target sequence, SWORD computes the diagonal where it is located, similarly to FASTA[3]. Target sequences are then selected based on the number of hits they have on the same diagonal. The pairwise alignment is then handled by the platform-accelerated Opal[4] library.
PySWRD is a Python module that provides bindings to the heuristic filter part of SWORD using Cython. It implements a user-friendly, Pythonic interface to build a heuristic filter, process a database in chunks, and produce the indices of targets passing the filter. The resulting indices can be used to filter a PyOpal database, using Opal for pairwise alignment like the original C++ implementation.
- no binary dependency: PySWRD is distributed as a Python package, so you can add it as a dependency to your project, and stop worrying about the SWORD binary being present on the end-user machine.
- no intermediate files: Everything happens in memory, in a Python object you control, so you don't have to invoke the SWORD CLI using a sub-process and temporary files.
- better portability: Using only the heuristic filter of SWORD allows the code to be independent of the local CPU features, unlike SWORD and Opal which require SIMD. PySWRD delegates the SIMD compilation and dynamic dispatch to PyOpal to make the package easier to install. It also benefits from the wider platform support of PyOpal compared to the original Opal, featuring support for Windows and for Aarch64 CPUs.
🔧 Installing
PySWRD is available for all modern Python versions (3.6+).
It can be installed directly from PyPI, which hosts some pre-built x86-64 wheels for Linux, MacOS, and Windows, as well as the code required to compile from source with Cython:
$ pip install pyswrd
💡 Example
PySWRD does not provide I/O, so the sequences to be used have to be loaded through another library, such as Biopython. PySWRD only requires the sequences to be available as Python strings:
targets = [
'MAFSAEDVLKEYDRRRRMEALLLSLYYPNDRKLLDYKEWSPPRVQVECPK',
'MSIIGATRLQNDKSDTYSAGPCYAGGCSAFTPRGTCGKDWDLGEQTCASG',
'MASNTVSAQGGSNRPVRDFSNIQDVAQFLLFDPIWNEQPGSIVPWKMNRE',
'MYQAINPCPQSWYGSPQLEREIVCKMSGAPHYPNYYPVHPNALGGAWFDT',
'MARPLLGKTSSVRRRLESLSACSIFFFLRKFCQKMASLVFLNSPVYQMSN'
]
queries = [
'MASNTVSAQGGSNRPVRDFSNIQDVAQFLLFDPIWNEQPG',
'MSFKVYDPIAELIATQFPTSNPDLQIINNDVLVVSPHKIT',
'MEQVPIKEMRLSDLRPNNKSIDTDLGGTKLVVIGKPGSGK'
]
Use the high-level search
function, which wraps the internal classes in a single
function to quickly run many-to-many searches in the event all your sequences are in
memory. It expects the sequences as iterable of Python strings, and yields hits
passing E-value and alignment thresholds:
import pyswrd
for hit in pyswrd.search(queries, targets):
print(hit.query_index, hit.target_index, hit.score, hit.evalue)
Different parameters can be passed to pyswrd.search
and are passed to the
SWORD filter and Opal alignment. For instance, to run SWORD in fast mode
instead of the default sensitive mode, and using the PAM70 matrix instead
of BLOSUM62, use:
for hit in pyswrd.search(queries, targets, scorer_name="PAM70", score_threshold=0, kmer_length=5):
print(hit.query_index, hit.target_index, hit.score, hit.evalue)
By default multithreading is supported, using one thread per CPU on the local
machine as reported by os.cpu_count
, but it can be changed with the threads
argument:
for hit in pyswrd.search(queries, targets, threads=1):
print(hit.query_index, hit.target_index, hit.score, hit.evalue)
You can also use the pyswrd.HeuristicFilter
class directly if you wish to
manage the data yourself, or if you want to use a different aligner.
⏱️ Benchmarks
The table below shows the time for running pyswrd.search
using 196 proteins
as queries (uniprot_sprot196.fasta
) against a database of 12,701 proteins
(uniprot_sprot12071.fasta
) pre-loaded into memory:
threads=1 |
threads=2 |
threads=4 |
threads=8 |
threads=12 |
|
---|---|---|---|---|---|
max_candidates=10 |
0.87s | 0.83s | 0.83s | 0.80s | 0.76s |
max_candidates=50 |
0.98s | 0.91s | 0.98s | 0.97s | 1.04s |
max_candidates=100 |
1.24s | 1.33s | 1.44s | 1.63s | 1.67s |
max_candidates=500 |
1.86s | 1.83s | 1.95s | 2.09s | 2.15s |
max_candidates=1000 |
2.87s | 2.64s | 2.83s | 2.82s | 2.90s |
max_candidates=5000 |
9.33s | 8.11s | 7.59s | 6.60s | 6.06s |
max_candidates=15000 |
21.50s | 15.85s | 14.74s | 11.83s | 11.34s |
max_candidates=30000 |
23.44s | 16.13s | 14.61s | 12.47s | 11.08s |
no filter (Opal) | 31.38s | 23.60s | 19.57s | 15.43s | 14.60s |
BLAST+ (blastp ) |
7.46s | 4.97s | 4.01s | 3.63s | 3.66s |
The max_candidates
parameter controls the strictness of the SWORD heuristic filter, and reduces
the total number of alignments made by Opal, at the cost of a lowered sensivity
(see SWORD Supplementary Figs. S1 and S2.).
SWORD uses 15,000 candidates in fast mode and 30,000 candidates in sensitive mode by default.
This was benchmarked against the NCBI NR
database, which contains more than 54M sequences; it is likely a smaller max_candidates
value can
be selected for smaller databases and/or databases with less redundant sequences without loss of sensitivity.
💭 Feedback
⚠️ Issue Tracker
Found a bug ? Have an enhancement request ? Head over to the GitHub issue tracker if you need to report or ask something. If you are filing in on a bug, please include as much information as you can about the issue, and try to recreate the same bug in a simple, easily reproducible situation.
🏗️ Contributing
Contributions are more than welcome! See
CONTRIBUTING.md
for more details.
📋 Changelog
This project adheres to Semantic Versioning and provides a changelog in the Keep a Changelog format.
⚖️ License
This library is provided under the GNU General Public License v3.0.
SWORD was written by Robert Vaser and is distributed under the terms of the
GPLv3 as well. See vendor/sword/LICENSE
for more information. SWORD redistributes additional
libraries under the terms of the MIT License.
This project is in no way not affiliated, sponsored, or otherwise endorsed by the SWORD authors. It was developed by Martin Larralde during his PhD project at the European Molecular Biology Laboratory in the Zeller team.
📚 References
- [1] Stephen F. Altschul, Warren Gish, Webb Miller, Eugene W. Myers, David J. Lipman. Basic local alignment search tool. J Mol Biol. 1990 Oct 5;215(3):403-10. doi:10.1016/S0022-2836(05)80360-2. PMID:2231712.
- [2] Robert Vaser, Dario Pavlović, Mile Šikić. SWORD—a highly efficient protein database search. Bioinformatics, Volume 32, Issue 17, September 2016, Pages i680–i684, doi:10.1093/bioinformatics/btw445.
- [3] David J. Lipman, William R. Pearson. Rapid and sensitive protein similarity searches. Science. 1985 Mar 22;227(4693):1435-41. doi:10.1126/science.2983426. PMID:2983426.
- [4] Korpar Matija, Martin Šošić, Dino Blažeka, Mile Šikić. SW#db: ‘GPU-Accelerated Exact Sequence Similarity Database Search’. PLoS One. 2015 Dec 31;10(12):e0145857. doi:10.1371/journal.pone.0145857. PMID:26719890. PMC4699916.
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 Distributions
Hashes for pyswrd-0.2.0-pp310-pypy310_pp73-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 871df8d4f9b7c3238b25be6c58908c68c61abfc239bf5e049e63969bef2e4cae |
|
MD5 | 3def512e88981d98d552d25771cc786c |
|
BLAKE2b-256 | 9c9edb9a93cbd6cfbb94a4a2f0b46ed02f517e2463755146bda53d17758cdf27 |
Hashes for pyswrd-0.2.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 79d0efbbc500467f049f7e65f648806cf8f183b7b71ae98632e35502af436247 |
|
MD5 | c53788ed81daef9a857e5acdc059adf7 |
|
BLAKE2b-256 | 54696afad6468dbb687036cef9578ba7237c4abeee74187e2dbe56ba35da28a2 |
Hashes for pyswrd-0.2.0-pp310-pypy310_pp73-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 850aa055960e1f2a860894baa59cd1c3bc3653c046ad5410f83cc7837f43d513 |
|
MD5 | 2dc5364a8548c5781550d206ef8ba897 |
|
BLAKE2b-256 | 1cf9ba7d611e8f587bae87a06349180d68e4c66fc6dbf77ba962b1d8f9946ca6 |
Hashes for pyswrd-0.2.0-pp39-pypy39_pp73-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | bc0dda4053e527ca9d1025198039a4d671fb3a0e8b16ef3b4b29b34b124e3152 |
|
MD5 | 5ee872759c95634adeeeda6303286be9 |
|
BLAKE2b-256 | c381e969086cad47ab902e3269e8f678170a29f85fad33cb00d07d8cfe814377 |
Hashes for pyswrd-0.2.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | ceb3b067b75f38180ebc8f2a8800500b14eb621347bec89cbdeafd6e52c107da |
|
MD5 | ac7dffc2d3fd446cd63fc680239d354d |
|
BLAKE2b-256 | 03b4cdb0c80beef4f01fd60f9211c80b5b88952911dddea69120c5ea215bb911 |
Hashes for pyswrd-0.2.0-pp39-pypy39_pp73-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | f4b3cbbee5a6bc8199256fe5641f337b6696cf334e61b1d20ace4bf778f7d21d |
|
MD5 | 6f1bf7b7e7c1e8d302596f537f3262c5 |
|
BLAKE2b-256 | 88de8cddee32552ece8a698e26732716d0ca27b65cfccf8aa428df158c0ccc37 |
Hashes for pyswrd-0.2.0-pp38-pypy38_pp73-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 944fb4556be4ccdf31fe473a84c02f72ee568b503f020088075ba3f48f651ff7 |
|
MD5 | 301d5cdd3e4e3b2acf3ee35180c84796 |
|
BLAKE2b-256 | d965659c5597d8013e1b49a4f04348f48099071eaae359b571e1f34a43b19120 |
Hashes for pyswrd-0.2.0-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | a2c738e5365c56b4f63f9925aef7d684e78bb06a3b35d39f50f954defa796727 |
|
MD5 | 294c2c0171fdd1fbb9c756d22beed698 |
|
BLAKE2b-256 | 1b7a4a60191d94e2e4446294aa9e8cb284c0df641c0c35fe706bde01222d3d57 |
Hashes for pyswrd-0.2.0-pp38-pypy38_pp73-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 81f4e8bc2b03d437ccfbb9c5c8a32aab70d6e1f17187fd4c23b6996205e50f2b |
|
MD5 | d38eb3c0d0145e6c6582640044ff3450 |
|
BLAKE2b-256 | 90e3c72ba89c3e38d090dc2cb40880f5b4d9b8c385037c566c399c5ae0397b65 |
Hashes for pyswrd-0.2.0-pp37-pypy37_pp73-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | c4970fe902b13fb1fd0924f50fae58e2608fdfc5b22dc568a8f7d44b1a2456fe |
|
MD5 | d0d5a137d0e8c030ac3df5593e11a541 |
|
BLAKE2b-256 | 310fc7e006c5514518fd51d98c75296fbec30101980c157ea6a96fef828f644d |
Hashes for pyswrd-0.2.0-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 00e144e9084a0ccf7305ef88a5d13b78e04596bd279ff896073d24a0f77729a6 |
|
MD5 | 5ce31d90380bbbbdc1c041e8f72b03e8 |
|
BLAKE2b-256 | 2961b969fdc6e7b34719d89bfd1b0d63dd11af8b3d27e538a8f0c318e609c127 |
Hashes for pyswrd-0.2.0-pp37-pypy37_pp73-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 504a675c6d8a0fe7061bf6e931dc125f2a45c65f0640657d03d8a43297004c72 |
|
MD5 | b00c77334eb0fe95df5859b2414655f9 |
|
BLAKE2b-256 | d8d78e3cf0051e8f3de4b86f66ecbe2f18e7673feb475051614a53c0f95a3a37 |
Hashes for pyswrd-0.2.0-cp312-cp312-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 84f0f447f5430e5a19416fe0e052409cada67b855240f8d2f4f3332777b265c9 |
|
MD5 | 5da343adf240797afc0de4aef1bc74a1 |
|
BLAKE2b-256 | 27208aff617582ccd32b37b3650b7049e6edee8d1b7ccb908280614d0d21bfac |
Hashes for pyswrd-0.2.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 0c51bb3e7f7f0d232b34cd8d98271d59bdb4252461e91fa32d9b8a5c565f85f8 |
|
MD5 | 074d16220d350c08c77651932bb1c252 |
|
BLAKE2b-256 | 7bbc8242d06404bb5100d527d65e8136151662ae28646eee9906720b75c13c93 |
Hashes for pyswrd-0.2.0-cp312-cp312-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 4fb661c60a169ba10c17e9bf3d8f8f14fa3dc01e11989cea4033d759a231e81b |
|
MD5 | e62500e91f59143cf30666ca30f8cfa3 |
|
BLAKE2b-256 | 29143334752f22234049adedcf83d0e4ce7e9d8040fe67a384859670689e08eb |
Hashes for pyswrd-0.2.0-cp311-cp311-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 08a3c011e85ffe8a2fd23901557866c38d54dc5ea17da9f5a94ca468662573f2 |
|
MD5 | bb0c3ad52d9cb986de626d8d1653c454 |
|
BLAKE2b-256 | a8a99c30acfc5f333c98e7b94ef58fbcf9c80482936e80e23240e06344eefa38 |
Hashes for pyswrd-0.2.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 345e862b3ab5d6ce80812d78792d0e536e60bee34e4e3e135788713af9cdfb36 |
|
MD5 | c1889a8ef776065c5208a1baeca803be |
|
BLAKE2b-256 | 7dd8d674ba5930e1e3ab91f7017e503574d0d0d782b292b1361d6ecf60b9d7d2 |
Hashes for pyswrd-0.2.0-cp311-cp311-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 60d131abd6faf2b9a6d197ba9768f0393366d28bf78f6770a21a42115ec7b2f9 |
|
MD5 | 84996de7768ba09156fdaefe5cd1fd14 |
|
BLAKE2b-256 | 5d52beb222241c7af7aeca4bfe9925a4c0adf561ae9b7573aba7b05935dfb13f |
Hashes for pyswrd-0.2.0-cp310-cp310-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 094097b9c9014ae30258cf9356743e7bd88918b92aa7ff1c27f9612777a4c0eb |
|
MD5 | df673881b5997762304164c5433a1378 |
|
BLAKE2b-256 | 19e9dd5cc5715a89cbe2c83fc57fe4e18ba490a3e7fd6ed95b086a3556f8d173 |
Hashes for pyswrd-0.2.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 4652241aae5d4c759ded00eec071492cf88ef0fe872c11c089aa475ecdc7caf7 |
|
MD5 | 5f778a29d222c27218f0121fb65e2889 |
|
BLAKE2b-256 | 966cf16ef958bfd09c3f5bb22205cfd40a7960f6bfa604b35bf52582073125d4 |
Hashes for pyswrd-0.2.0-cp310-cp310-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | ba35db937ba6800bbb6d5170ff121a3860fb3f1a80813d8b7259567795f39020 |
|
MD5 | fbc502e1cb4aabf7ce65a28270ecbf64 |
|
BLAKE2b-256 | 6d69ff7008f058f4cf8633f3dcdfb85fdb1a6db01b75f988b1fc28b6c30edd75 |
Hashes for pyswrd-0.2.0-cp39-cp39-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | ee816d3a3b9016f787c8335c3b35f05c73203d40ce9d653aae43d199e86d77fa |
|
MD5 | 35d091b67015b4689c49ee45c8f5e73e |
|
BLAKE2b-256 | 16fe9d569a3e243a2d05285096d5658d2358ea48aa86f20477e3194cb267505c |
Hashes for pyswrd-0.2.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 1f38a41fb54ab919f72c492975b405330f7ae6ea9a0089f2af013b449d896fef |
|
MD5 | f1667a96cab95d764687679955093ba6 |
|
BLAKE2b-256 | 4b81fedcfeae78fe40d3e5977a0d9f39ae88140bd8f2a9d55f347e8257cbf0e5 |
Hashes for pyswrd-0.2.0-cp39-cp39-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | b54f8de616a93c028d7cfe4a76d45a05c398605608184cc57355455d9b92503b |
|
MD5 | 42bc868aad7b2c4bc3e70fe30a2827eb |
|
BLAKE2b-256 | 0e4dc68e61d95f352250c4c3052de04ca2aff8b7d614f7602e689d665d761521 |
Hashes for pyswrd-0.2.0-cp38-cp38-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | e878021f03f5445674e017c9950c4f5cb11d40aaa5af9a368bac4170b07b2061 |
|
MD5 | cabdc9bcd76a8890ba35d915759b07e8 |
|
BLAKE2b-256 | 0841d68c26ebcffc5ae40c4582940bc8d7f16609586e55ff36dbbe091c253b7a |
Hashes for pyswrd-0.2.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 2e6b8a076379b4a6e744f2a1090496b4483745b5d1268a6ab9a366c5d41d3b5b |
|
MD5 | 337f465902c1cf8884cc6be895e2f773 |
|
BLAKE2b-256 | a9a87b190593ce269f2635afdf6dd57015c8f8832f036ae5ce567f452ef93353 |
Hashes for pyswrd-0.2.0-cp38-cp38-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 601aea1d3dabe70d3379ce258b26dfb6e864739e54d4352e0f142c02db37ce27 |
|
MD5 | a2388bed6a782262cbe4dcc7967bc476 |
|
BLAKE2b-256 | 33f84ae4c709fd7f7f8ae0398549b284357733dc554526b989555ef041740a1f |
Hashes for pyswrd-0.2.0-cp37-cp37m-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | ed94805aeab59b3be6c24fcda3e154bcb219ac74322c2e33238a72064ff6effe |
|
MD5 | 9d5f0664304fd853c207fe241fc8dcce |
|
BLAKE2b-256 | 17261c4286f8806b5f1ac8a0ab5a180c79ce61b3c93b9fccb3083b8ef9e337ae |
Hashes for pyswrd-0.2.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | b052b2b9deb6c9158dd2b98308e7dddc7a2f7816b8052eb0eb114e7f7ba45024 |
|
MD5 | 8cd6e163ff8651409ea2a050945189c5 |
|
BLAKE2b-256 | a045c508e94eec47b644d5b1053efe90c8567a67117f04af48347912df0a96a4 |
Hashes for pyswrd-0.2.0-cp37-cp37m-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 01e250e0f0bf25ba5b997e9a6f74a80710f4ae3290c8f69ab29fac79984ed7a4 |
|
MD5 | a3d86165f05a94da90088e929b166a08 |
|
BLAKE2b-256 | 197a45a51891d0e370a6220aa2764cc0a5bf198bd25c3524827592c0e5105174 |
Hashes for pyswrd-0.2.0-cp36-cp36m-win_amd64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 558d38b1c5b85648ca9d9e98fa1a0504b44d4d9c1bf8b72475204dafb8ae88aa |
|
MD5 | 28c4467b796bb7d6816dd96274f5f68e |
|
BLAKE2b-256 | 5b9fc7c0732ca857b81509c1b373321bbcacbbaa579f045e1c338bfbb092a3dd |
Hashes for pyswrd-0.2.0-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | a821a68c1a9040f02ab20e278413f0013e5ac9802e1358f26dcc00700fa92a54 |
|
MD5 | 945d5f07bc4fe3edfa49e49789dca229 |
|
BLAKE2b-256 | be84da40b6fcb5f6bbf53c4205b590adc3e060016120a111b836315323af11a7 |
Hashes for pyswrd-0.2.0-cp36-cp36m-macosx_10_12_x86_64.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | b995b19e1472e24964c42da59fd3e015fc952bf6e91eebe9080e98a8f401e8d8 |
|
MD5 | e52834dbc86507fc322a2efab20e8355 |
|
BLAKE2b-256 | 0db416d94eb6cd08eb3581c1d244d10d762ff176c3e6e728af5dd848cdf306c6 |