%global _empty_manifest_terminate_build 0 Name: python-ahocorapy Version: 1.6.2 Release: 1 Summary: ahocorapy - Pure python ahocorasick implementation License: MIT URL: https://github.com/abusix/ahocorapy Source0: https://mirrors.nju.edu.cn/pypi/web/packages/38/d7/81b1d1533896d72186add63e3dd2fb70142aede6cc8a3cc48bf8a6d51002/ahocorapy-1.6.2.tar.gz BuildArch: noarch Requires: python3-future %description [![Test](https://img.shields.io/github/workflow/status/abusix/ahocorapy/test/master)](https://github.com/abusix/ahocorapy/actions) [![Test Coverage](https://img.shields.io/codecov/c/gh/abusix/ahocorapy/master)](https://codecov.io/gh/abusix/ahocorapy) [![Downloads](https://pepy.tech/badge/ahocorapy)](https://pepy.tech/project/ahocorapy) [![PyPi Version](https://img.shields.io/pypi/v/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) [![PyPi License](https://img.shields.io/pypi/l/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) [![PyPi Versions](https://img.shields.io/pypi/pyversions/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) [![PyPi Wheel](https://img.shields.io/pypi/wheel/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) # ahocorapy - Fast Many-Keyword Search in Pure Python ahocorapy is a pure python implementation of the Aho-Corasick Algorithm. Given a list of keywords one can check if at least one of the keywords exist in a given text in linear time. ## Comparison: ### Why another Aho-Corasick implementation? We started working on this in the beginning of 2016. Our requirements included unicode support combined with python2.7. That was impossible with C-extension based libraries (like [pyahocorasick](https://github.com/WojciechMula/pyahocorasick/)). Pure python libraries were very slow or unusable due to memory explosion. Since then another pure python library was released [py-aho-corasick](https://github.com/JanFan/py-aho-corasick). The repository also contains some discussion about different implementations. There is also [acora](https://github.com/scoder/acora), but it includes the note ('current construction algorithm is not suitable for really large sets of keywords') which really was the case the last time I tested, because RAM ran out quickly. ### Differences - Compared to [pyahocorasick](https://github.com/WojciechMula/pyahocorasick/) our library supports unicode in python 2.7 just like [py-aho-corasick](https://github.com/JanFan/py-aho-corasick). We don't use any C-Extension so the library is not platform dependant. - On top of the standard Aho-Corasick longest suffix search, we also perform a shortcutting routine in the end, so that our lookup is fast while, the setup takes longer. During set up we go through the states and directly add transitions that are "offered" by the longest suffix or their longest suffixes. This leads to faster lookup times, because in the end we only have to follow simple transitions and don't have to perform any additional suffix lookup. It also leads to a bigger memory footprint, because the number of transitions is higher, because they are all included explicitely and not implicitely hidden by suffix pointers. - We added a small tool that helps you visualize the resulting graph. This may help understanding the algorithm, if you'd like. See below. - Fully pickleable (pythons built-in de-/serialization). ahocorapy uses a non-recursive custom implementation for de-/serialization so that even huge keyword trees can be pickled. ### Performance I compared the two libraries mentioned above with ahocorapy. We used 50,000 keywords long list and an input text of 34,199 characters. In the text only one keyword of the list is contained. The setup process was run once per library and the search process was run 100 times. The following results are in seconds (not averaged for the lookup). You can perform this test yourself using `python tests/ahocorapy_performance_test.py`. (Except for the pyahocorasick_py results. These were taken by importing the pure python version of the code of [pyahocorasick](https://github.com/WojciechMula/pyahocorasick/). It's not available through pypi as stated in the code.) I also added measurements for the pure python libraries with run with pypy. These are the results: | Library (Variant) | Setup (1x) | Search (100x) | | ------------------------------------------------------ | ---------- | ------------- | | ahocorapy\* | 0.30s | 0.29s | | ahocorapy (run with pypy)\* | 0.37s | 0.10s | | pyahocorasick\* | 0.04s | 0.04s | | pyahocorasick (run with pypy)\* | 0.10s | 0.05s | | pyahocorasick (pure python variant in github repo)\*\* | 0.50s | 1.68s | | py_aho_corasick\* | 0.72s | 4,60s | | py_aho_corasick (run with pypy)\* | 0.83s | 2.02s | As expected the C-Extension shatters the pure python implementations. Even though there is probably still room for optimization in ahocorapy we are not going to get to the mark that pyahocorasick sets. ahocorapy's lookups are faster than py_aho_corasick. When run with pypy ahocorapy is almost as fast as pyahocorasick, at least when it comes to searching. The setup overhead is higher due to the suffix shortcutting mechanism used. \* Specs CPU: AMD Ryzen 2700X Linux Kernel: 6.0.6 CPython: 3.11.0 pypy: PyPy 7.3.9 (Python 3.9.12) with GCC 10.2.1 20210130 Date tested: 2022-11-22 \*\* Old measurement with different specs ## Basic Usage: ### Installation ``` pip install ahocorapy ``` ### Creation of the Search Tree ```python from ahocorapy.keywordtree import KeywordTree kwtree = KeywordTree(case_insensitive=True) kwtree.add('malaga') kwtree.add('lacrosse') kwtree.add('mallorca') kwtree.add('mallorca bella') kwtree.add('orca') kwtree.finalize() ``` ### Searching ```python result = kwtree.search('My favorite islands are malaga and sylt.') print(result) ``` Prints : ```python ('malaga', 24) ``` The search_all method returns a generator for all keywords found, or None if there is none. ```python results = kwtree.search_all('malheur on mallorca bellacrosse') for result in results: print(result) ``` Prints : ```python ('mallorca', 11) ('orca', 15) ('mallorca bella', 11) ('lacrosse', 23) ``` ### Thread Safety The construction of the tree is currently NOT thread safe. That means `add`ing shouldn't be called multiple times concurrently. Behavior is undefined. After `finalize` is called you can use the `search` functionality on the same tree from multiple threads at the same time. So that part is thread safe. ## Drawing Graph You can print the underlying graph with the Visualizer class. This feature requires a working pygraphviz library installed. ```python from ahocorapy_visualizer.visualizer import Visualizer visualizer = Visualizer() visualizer.draw('readme_example.png', kwtree) ``` The resulting .png of the graph looks like this: ![graph for kwtree](https://raw.githubusercontent.com/abusix/ahocorapy/master/img/readme_example.png "Keyword Tree") %package -n python3-ahocorapy Summary: ahocorapy - Pure python ahocorasick implementation Provides: python-ahocorapy BuildRequires: python3-devel BuildRequires: python3-setuptools BuildRequires: python3-pip %description -n python3-ahocorapy [![Test](https://img.shields.io/github/workflow/status/abusix/ahocorapy/test/master)](https://github.com/abusix/ahocorapy/actions) [![Test Coverage](https://img.shields.io/codecov/c/gh/abusix/ahocorapy/master)](https://codecov.io/gh/abusix/ahocorapy) [![Downloads](https://pepy.tech/badge/ahocorapy)](https://pepy.tech/project/ahocorapy) [![PyPi Version](https://img.shields.io/pypi/v/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) [![PyPi License](https://img.shields.io/pypi/l/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) [![PyPi Versions](https://img.shields.io/pypi/pyversions/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) [![PyPi Wheel](https://img.shields.io/pypi/wheel/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) # ahocorapy - Fast Many-Keyword Search in Pure Python ahocorapy is a pure python implementation of the Aho-Corasick Algorithm. Given a list of keywords one can check if at least one of the keywords exist in a given text in linear time. ## Comparison: ### Why another Aho-Corasick implementation? We started working on this in the beginning of 2016. Our requirements included unicode support combined with python2.7. That was impossible with C-extension based libraries (like [pyahocorasick](https://github.com/WojciechMula/pyahocorasick/)). Pure python libraries were very slow or unusable due to memory explosion. Since then another pure python library was released [py-aho-corasick](https://github.com/JanFan/py-aho-corasick). The repository also contains some discussion about different implementations. There is also [acora](https://github.com/scoder/acora), but it includes the note ('current construction algorithm is not suitable for really large sets of keywords') which really was the case the last time I tested, because RAM ran out quickly. ### Differences - Compared to [pyahocorasick](https://github.com/WojciechMula/pyahocorasick/) our library supports unicode in python 2.7 just like [py-aho-corasick](https://github.com/JanFan/py-aho-corasick). We don't use any C-Extension so the library is not platform dependant. - On top of the standard Aho-Corasick longest suffix search, we also perform a shortcutting routine in the end, so that our lookup is fast while, the setup takes longer. During set up we go through the states and directly add transitions that are "offered" by the longest suffix or their longest suffixes. This leads to faster lookup times, because in the end we only have to follow simple transitions and don't have to perform any additional suffix lookup. It also leads to a bigger memory footprint, because the number of transitions is higher, because they are all included explicitely and not implicitely hidden by suffix pointers. - We added a small tool that helps you visualize the resulting graph. This may help understanding the algorithm, if you'd like. See below. - Fully pickleable (pythons built-in de-/serialization). ahocorapy uses a non-recursive custom implementation for de-/serialization so that even huge keyword trees can be pickled. ### Performance I compared the two libraries mentioned above with ahocorapy. We used 50,000 keywords long list and an input text of 34,199 characters. In the text only one keyword of the list is contained. The setup process was run once per library and the search process was run 100 times. The following results are in seconds (not averaged for the lookup). You can perform this test yourself using `python tests/ahocorapy_performance_test.py`. (Except for the pyahocorasick_py results. These were taken by importing the pure python version of the code of [pyahocorasick](https://github.com/WojciechMula/pyahocorasick/). It's not available through pypi as stated in the code.) I also added measurements for the pure python libraries with run with pypy. These are the results: | Library (Variant) | Setup (1x) | Search (100x) | | ------------------------------------------------------ | ---------- | ------------- | | ahocorapy\* | 0.30s | 0.29s | | ahocorapy (run with pypy)\* | 0.37s | 0.10s | | pyahocorasick\* | 0.04s | 0.04s | | pyahocorasick (run with pypy)\* | 0.10s | 0.05s | | pyahocorasick (pure python variant in github repo)\*\* | 0.50s | 1.68s | | py_aho_corasick\* | 0.72s | 4,60s | | py_aho_corasick (run with pypy)\* | 0.83s | 2.02s | As expected the C-Extension shatters the pure python implementations. Even though there is probably still room for optimization in ahocorapy we are not going to get to the mark that pyahocorasick sets. ahocorapy's lookups are faster than py_aho_corasick. When run with pypy ahocorapy is almost as fast as pyahocorasick, at least when it comes to searching. The setup overhead is higher due to the suffix shortcutting mechanism used. \* Specs CPU: AMD Ryzen 2700X Linux Kernel: 6.0.6 CPython: 3.11.0 pypy: PyPy 7.3.9 (Python 3.9.12) with GCC 10.2.1 20210130 Date tested: 2022-11-22 \*\* Old measurement with different specs ## Basic Usage: ### Installation ``` pip install ahocorapy ``` ### Creation of the Search Tree ```python from ahocorapy.keywordtree import KeywordTree kwtree = KeywordTree(case_insensitive=True) kwtree.add('malaga') kwtree.add('lacrosse') kwtree.add('mallorca') kwtree.add('mallorca bella') kwtree.add('orca') kwtree.finalize() ``` ### Searching ```python result = kwtree.search('My favorite islands are malaga and sylt.') print(result) ``` Prints : ```python ('malaga', 24) ``` The search_all method returns a generator for all keywords found, or None if there is none. ```python results = kwtree.search_all('malheur on mallorca bellacrosse') for result in results: print(result) ``` Prints : ```python ('mallorca', 11) ('orca', 15) ('mallorca bella', 11) ('lacrosse', 23) ``` ### Thread Safety The construction of the tree is currently NOT thread safe. That means `add`ing shouldn't be called multiple times concurrently. Behavior is undefined. After `finalize` is called you can use the `search` functionality on the same tree from multiple threads at the same time. So that part is thread safe. ## Drawing Graph You can print the underlying graph with the Visualizer class. This feature requires a working pygraphviz library installed. ```python from ahocorapy_visualizer.visualizer import Visualizer visualizer = Visualizer() visualizer.draw('readme_example.png', kwtree) ``` The resulting .png of the graph looks like this: ![graph for kwtree](https://raw.githubusercontent.com/abusix/ahocorapy/master/img/readme_example.png "Keyword Tree") %package help Summary: Development documents and examples for ahocorapy Provides: python3-ahocorapy-doc %description help [![Test](https://img.shields.io/github/workflow/status/abusix/ahocorapy/test/master)](https://github.com/abusix/ahocorapy/actions) [![Test Coverage](https://img.shields.io/codecov/c/gh/abusix/ahocorapy/master)](https://codecov.io/gh/abusix/ahocorapy) [![Downloads](https://pepy.tech/badge/ahocorapy)](https://pepy.tech/project/ahocorapy) [![PyPi Version](https://img.shields.io/pypi/v/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) [![PyPi License](https://img.shields.io/pypi/l/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) [![PyPi Versions](https://img.shields.io/pypi/pyversions/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) [![PyPi Wheel](https://img.shields.io/pypi/wheel/ahocorapy.svg)](https://pypi.python.org/pypi/ahocorapy) # ahocorapy - Fast Many-Keyword Search in Pure Python ahocorapy is a pure python implementation of the Aho-Corasick Algorithm. Given a list of keywords one can check if at least one of the keywords exist in a given text in linear time. ## Comparison: ### Why another Aho-Corasick implementation? We started working on this in the beginning of 2016. Our requirements included unicode support combined with python2.7. That was impossible with C-extension based libraries (like [pyahocorasick](https://github.com/WojciechMula/pyahocorasick/)). Pure python libraries were very slow or unusable due to memory explosion. Since then another pure python library was released [py-aho-corasick](https://github.com/JanFan/py-aho-corasick). The repository also contains some discussion about different implementations. There is also [acora](https://github.com/scoder/acora), but it includes the note ('current construction algorithm is not suitable for really large sets of keywords') which really was the case the last time I tested, because RAM ran out quickly. ### Differences - Compared to [pyahocorasick](https://github.com/WojciechMula/pyahocorasick/) our library supports unicode in python 2.7 just like [py-aho-corasick](https://github.com/JanFan/py-aho-corasick). We don't use any C-Extension so the library is not platform dependant. - On top of the standard Aho-Corasick longest suffix search, we also perform a shortcutting routine in the end, so that our lookup is fast while, the setup takes longer. During set up we go through the states and directly add transitions that are "offered" by the longest suffix or their longest suffixes. This leads to faster lookup times, because in the end we only have to follow simple transitions and don't have to perform any additional suffix lookup. It also leads to a bigger memory footprint, because the number of transitions is higher, because they are all included explicitely and not implicitely hidden by suffix pointers. - We added a small tool that helps you visualize the resulting graph. This may help understanding the algorithm, if you'd like. See below. - Fully pickleable (pythons built-in de-/serialization). ahocorapy uses a non-recursive custom implementation for de-/serialization so that even huge keyword trees can be pickled. ### Performance I compared the two libraries mentioned above with ahocorapy. We used 50,000 keywords long list and an input text of 34,199 characters. In the text only one keyword of the list is contained. The setup process was run once per library and the search process was run 100 times. The following results are in seconds (not averaged for the lookup). You can perform this test yourself using `python tests/ahocorapy_performance_test.py`. (Except for the pyahocorasick_py results. These were taken by importing the pure python version of the code of [pyahocorasick](https://github.com/WojciechMula/pyahocorasick/). It's not available through pypi as stated in the code.) I also added measurements for the pure python libraries with run with pypy. These are the results: | Library (Variant) | Setup (1x) | Search (100x) | | ------------------------------------------------------ | ---------- | ------------- | | ahocorapy\* | 0.30s | 0.29s | | ahocorapy (run with pypy)\* | 0.37s | 0.10s | | pyahocorasick\* | 0.04s | 0.04s | | pyahocorasick (run with pypy)\* | 0.10s | 0.05s | | pyahocorasick (pure python variant in github repo)\*\* | 0.50s | 1.68s | | py_aho_corasick\* | 0.72s | 4,60s | | py_aho_corasick (run with pypy)\* | 0.83s | 2.02s | As expected the C-Extension shatters the pure python implementations. Even though there is probably still room for optimization in ahocorapy we are not going to get to the mark that pyahocorasick sets. ahocorapy's lookups are faster than py_aho_corasick. When run with pypy ahocorapy is almost as fast as pyahocorasick, at least when it comes to searching. The setup overhead is higher due to the suffix shortcutting mechanism used. \* Specs CPU: AMD Ryzen 2700X Linux Kernel: 6.0.6 CPython: 3.11.0 pypy: PyPy 7.3.9 (Python 3.9.12) with GCC 10.2.1 20210130 Date tested: 2022-11-22 \*\* Old measurement with different specs ## Basic Usage: ### Installation ``` pip install ahocorapy ``` ### Creation of the Search Tree ```python from ahocorapy.keywordtree import KeywordTree kwtree = KeywordTree(case_insensitive=True) kwtree.add('malaga') kwtree.add('lacrosse') kwtree.add('mallorca') kwtree.add('mallorca bella') kwtree.add('orca') kwtree.finalize() ``` ### Searching ```python result = kwtree.search('My favorite islands are malaga and sylt.') print(result) ``` Prints : ```python ('malaga', 24) ``` The search_all method returns a generator for all keywords found, or None if there is none. ```python results = kwtree.search_all('malheur on mallorca bellacrosse') for result in results: print(result) ``` Prints : ```python ('mallorca', 11) ('orca', 15) ('mallorca bella', 11) ('lacrosse', 23) ``` ### Thread Safety The construction of the tree is currently NOT thread safe. That means `add`ing shouldn't be called multiple times concurrently. Behavior is undefined. After `finalize` is called you can use the `search` functionality on the same tree from multiple threads at the same time. So that part is thread safe. ## Drawing Graph You can print the underlying graph with the Visualizer class. This feature requires a working pygraphviz library installed. ```python from ahocorapy_visualizer.visualizer import Visualizer visualizer = Visualizer() visualizer.draw('readme_example.png', kwtree) ``` The resulting .png of the graph looks like this: ![graph for kwtree](https://raw.githubusercontent.com/abusix/ahocorapy/master/img/readme_example.png "Keyword Tree") %prep %autosetup -n ahocorapy-1.6.2 %build %py3_build %install %py3_install install -d -m755 %{buildroot}/%{_pkgdocdir} if [ -d doc ]; then cp -arf doc %{buildroot}/%{_pkgdocdir}; fi if [ -d docs ]; then cp -arf docs %{buildroot}/%{_pkgdocdir}; fi if [ -d example ]; then cp -arf example %{buildroot}/%{_pkgdocdir}; fi if [ -d examples ]; then cp -arf examples %{buildroot}/%{_pkgdocdir}; fi pushd %{buildroot} if [ -d usr/lib ]; then find usr/lib -type f -printf "/%h/%f\n" >> filelist.lst fi if [ -d usr/lib64 ]; then find usr/lib64 -type f -printf "/%h/%f\n" >> filelist.lst fi if [ -d usr/bin ]; then find usr/bin -type f -printf "/%h/%f\n" >> filelist.lst fi if [ -d usr/sbin ]; then find usr/sbin -type f -printf "/%h/%f\n" >> filelist.lst fi touch doclist.lst if [ -d usr/share/man ]; then find usr/share/man -type f -printf "/%h/%f.gz\n" >> doclist.lst fi popd mv %{buildroot}/filelist.lst . mv %{buildroot}/doclist.lst . %files -n python3-ahocorapy -f filelist.lst %dir %{python3_sitelib}/* %files help -f doclist.lst %{_docdir}/* %changelog * Fri Apr 21 2023 Python_Bot - 1.6.2-1 - Package Spec generated