Evaluation#
Although text is ubiquitous on the Web, extracting information from web pages can prove to be difficult. Should the tooling be adapted to particular news outlets or blogs that are targeted (which often amounts to the development of web scraping tools) or should the extraction be as generic as possible to provide opportunistic ways of gathering information?
The extraction focuses on the main content, which is usually the part displayed centrally, without the left or right bars, the header or the footer, but including potential titles and (optionally) comments. This task is also known as web scraping, boilerplate removal, DOM-based content extraction, main content identification, or web page cleaning.
Alternatives#
Although a few corresponding Python packages are not actively maintained the following alternatives exist.
These packages keep the structure intact but do not focus on main text extraction:
html2text converts HTML pages to Markup language
html_text converts HTML code to plain text
inscriptis converts HTML to text with a particular emphasis on nested tables
These packages focus on main text extraction:
boilerpy3 is a Python version of the boilerpipe algorithm for boilerplate removal and fulltext extraction
dragnet is not maintained anymore, it is provided for reference only (in older evaluations)
goose3 can extract information for embedded content but doesn’t preserve markup
jusText is designed to preserve mainly text containing full sentences along with some markup, it has been explicitly developed to create linguistic resources
newspaper3k is mostly geared towards newspaper texts, provides additional functions but no structured text or comment extraction
news-please is a news crawler that extracts structured information
readability-lxml cleans the page and preserves some markup
readabilipy contains a Python wrapper for Mozilla’s Node.js package, as well as article extraction routines written in pure Python
trafilatura is the library documented here, several options are tested regarding main text extraction only, without metadata or comments
The tools are compared to the raw page source and to a meaningful baseline consisting of extracting the raw text contained in the JSON article element or in a combination of paragraph, code and quote elements.
Description#
Test set: The experiments below are run on a collection of documents which are either typical for Internet articles (news outlets, blogs). Some are harder to process due to mixed content (lists, tables) or not fully valid HTML code. They are selected from large collections of web pages in German, for the sake of completeness documents in other languages are added (notably English, French, other European languages, Chinese and Arabic, about 20-30% of the total).
Evaluation: Decisive document segments are singled out which are not statistically representative but very significant in the perspective of working with the texts, most notably left/right columns, additional header, author or footer information such as imprints or addresses, as well as affiliated and social network links, in short boilerplate. Raw text segments are expected which is also a way to evaluate the quality of HTML extraction in itself.
Time: The execution time is provided as an indication. As the baseline extraction is simple and fast, it is used for the benchmark. Certain packages are noticeably slower than the rest: goose3 and newspaper, while news-please’s execution time isn’t comparable because of operations unrelated to text extraction. ReadabiliPy is very slow for unclear reasons.
Errors: The boilerpy3, newspaper3k, and readabilipy modules do not work without errors on every HTML file in the test set, probably because of malformed HTML, encoding or parsing bugs. These errors are ignored in order to complete the benchmark.
Results: The baseline beats a few systems, showing its interest. justext is highly configurable and tweaking its configuration (as it is done here) can lead to better performance than its generic settings. goose3 is the most precise algorithm, albeit at a significant cost in terms of recall. The packages focusing on raw text extraction html_text and inscriptis are roughly comparable and achieve the best recall as they try to extract all the text. Rule-based approaches such as trafilatura’s obtain balanced results despite a lack of precision. Combined with an algorithmic approach they perform significantly better than the other tested solutions.
Roadmap: Further evaluations will be run, including additional tools and languages. Comment extraction still has to be evaluated, although most libraries don not offer this functionality.
The evaluation script is available on the project repository: tests/README.rst. To reproduce the tests just clone the repository, install all necessary packages and run the evaluation script with the data provided in the tests directory.
Results (2022-05-18)#
750 documents, 2236 text & 2250 boilerplate segments, Python 3.8 |
|||||
---|---|---|---|---|---|
Python Package |
Precision |
Recall |
Accuracy |
F-Score |
Diff. |
raw HTML |
0.527 |
0.874 |
0.546 |
0.658 |
0 |
html2text 2020.1.16 |
0.486 |
0.709 |
0.481 |
0.577 |
7.6x |
html_text 0.5.2 |
0.529 |
0.958 |
0.554 |
0.682 |
2.2x |
inscriptis 2.2.0 (html to txt) |
0.534 |
0.959 |
0.563 |
0.686 |
3.5x |
newspaper3k 0.2.8 |
0.895 |
0.593 |
0.762 |
0.713 |
12x |
justext 3.0.0 (custom) |
0.865 |
0.650 |
0.775 |
0.742 |
5.2x |
boilerpy3 1.0.6 (article mode) |
0.814 |
0.744 |
0.787 |
0.777 |
4.1x |
baseline (text markup) |
0.757 |
0.827 |
0.781 |
0.790 |
1x |
goose3 3.1.9 |
0.934 |
0.690 |
0.821 |
0.793 |
22x |
readability-lxml 0.8.1 |
0.891 |
0.729 |
0.820 |
0.801 |
5.8x |
news-please 1.5.22 |
0.898 |
0.734 |
0.826 |
0.808 |
61x |
readabilipy 0.2.0 |
0.877 |
0.870 |
0.874 |
0.874 |
248x |
trafilatura 1.2.2 (fast) |
0.914 |
0.886 |
0.902 |
0.900 |
4.8x |
trafilatura 1.2.2 (precision) |
0.932 |
0.874 |
0.905 |
0.902 |
9.4x |
trafilatura 1.2.2 (standard) |
0.914 |
0.904 |
0.910 |
0.909 |
7.1x |
External evaluations#
Trafilatura is the most efficient open-source library in ScrapingHub’s article extraction benchmark.
Best overall tool according to Gaël Lejeune & Adrien Barbaresi, Bien choisir son outil d’extraction de contenu à partir du Web (2020, PDF, in French).
Comparison on a small sample of Polish news texts and forums.
Older results#
Older results (2021-06-07)#
500 documents, 1487 text and 1496 boilerplate segments |
|||||
---|---|---|---|---|---|
Python Package |
Precision |
Recall |
Accuracy |
F-Score |
Diff. |
raw HTML |
0.527 |
0.878 |
0.547 |
0.659 |
0 |
html2text 2020.1.16 |
0.488 |
0.714 |
0.484 |
0.580 |
8.9x |
html_text 0.5.2 |
0.526 |
0.958 |
0.548 |
0.679 |
1.9x |
inscriptis 1.1 (html to txt) |
0.531 |
0.958 |
0.556 |
0.683 |
2.4x |
justext 2.2.0 (custom) |
0.870 |
0.584 |
0.749 |
0.699 |
6.1x |
newspaper3k 0.2.8 |
0.921 |
0.574 |
0.763 |
0.708 |
12.9x |
boilerpy3 1.0.2 (article mode) |
0.851 |
0.696 |
0.788 |
0.766 |
4.8x |
goose3 3.1.9 |
0.950 |
0.644 |
0.806 |
0.767 |
18.8x |
baseline (text markup) |
0.746 |
0.804 |
0.766 |
0.774 |
1x |
dragnet 2.0.4 |
0.906 |
0.689 |
0.810 |
0.783 |
3.1x |
readability-lxml 0.8.1 |
0.917 |
0.716 |
0.826 |
0.804 |
5.9x |
news-please 1.5.21 |
0.924 |
0.718 |
0.830 |
0.808 |
60x |
trafilatura 0.8.2 (fast) |
0.925 |
0.868 |
0.899 |
0.896 |
3.9x |
trafilatura 0.8.2 |
0.934 |
0.890 |
0.914 |
0.912 |
8.4x |
Older results (2020-11-06)#
500 documents, 1487 text and 1496 boilerplate segments |
|||||
---|---|---|---|---|---|
Python Package |
Precision |
Recall |
Accuracy |
F-Score |
Diff. |
raw HTML |
0.527 |
0.878 |
0.547 |
0.659 |
0 |
html2text 2020.1.16 |
0.488 |
0.714 |
0.484 |
0.580 |
8.9x |
html_text 0.5.2 |
0.526 |
0.958 |
0.548 |
0.679 |
1.9x |
inscriptis 1.1 (html to txt) |
0.531 |
0.958 |
0.556 |
0.683 |
2.4x |
justext 2.2.0 (tweaked) |
0.870 |
0.584 |
0.749 |
0.699 |
6.1x |
newspaper3k 0.2.8 |
0.921 |
0.574 |
0.763 |
0.708 |
12.9x |
goose3 3.1.6 |
0.950 |
0.629 |
0.799 |
0.757 |
19.0x |
boilerpy3 1.0.2 (article mode) |
0.851 |
0.696 |
0.788 |
0.766 |
4.8x |
baseline (text markup) |
0.746 |
0.804 |
0.766 |
0.774 |
1x |
dragnet 2.0.4 |
0.906 |
0.689 |
0.810 |
0.783 |
3.1x |
readability-lxml 0.8.1 |
0.917 |
0.716 |
0.826 |
0.804 |
5.9x |
news-please 1.5.13 |
0.923 |
0.711 |
0.827 |
0.804 |
184x |
trafilatura 0.6.0 |
0.924 |
0.849 |
0.890 |
0.885 |
3.9x |
trafilatura 0.6.0 (+ fallbacks) |
0.933 |
0.877 |
0.907 |
0.904 |
8.4x |
Older results (2020-07-16)#
400 documents, 1186 text and 1198 boilerplate segments |
|||||
---|---|---|---|---|---|
Python Package |
Precision |
Recall |
Accuracy |
F-Score |
Diff. |
raw HTML |
0.524 |
0.879 |
0.543 |
0.657 |
0 |
html2text 2020.1.16 |
0.485 |
0.718 |
0.480 |
0.579 |
8.4x |
html_text 0.5.1 |
0.521 |
0.962 |
0.542 |
0.676 |
1.8x |
inscriptis 1.0 (html to txt) |
0.527 |
0.965 |
0.551 |
0.681 |
1.9x |
newspaper3k 0.2.8 |
0.916 |
0.577 |
0.763 |
0.708 |
11.8x |
justext 2.2.0 (tweaked) |
0.867 |
0.651 |
0.777 |
0.744 |
4.9x |
goose3 3.1.6 |
0.953 |
0.635 |
0.803 |
0.762 |
17.3x |
baseline (text markup) |
0.738 |
0.804 |
0.760 |
0.770 |
1x |
boilerpy3 1.0.2 (article mode) |
0.847 |
0.711 |
0.792 |
0.773 |
4.4x |
dragnet 2.0.4 |
0.906 |
0.704 |
0.816 |
0.792 |
2.8x |
readability-lxml 0.8.1 |
0.913 |
0.739 |
0.835 |
0.817 |
5.4x |
news-please 1.4.25 |
0.918 |
0.739 |
0.837 |
0.819 |
56.4x |
trafilatura 0.5.1 |
0.927 |
0.854 |
0.894 |
0.889 |
3.1x |
trafilatura 0.5.1 (+ fallbacks) |
0.933 |
0.885 |
0.911 |
0.908 |
6.8x |
Older results (2020-03-19)#
300 documents, 869 text and 878 boilerplate segments |
|||||
---|---|---|---|---|---|
Python Package |
Precision |
Recall |
Accuracy |
F-Score |
Time |
raw HTML |
0.519 |
0.885 |
0.535 |
0.654 |
0 |
baseline (text markup) |
0.726 |
0.776 |
0.742 |
0.750 |
1.14 |
html2text 2020.1.16 |
0.499 |
0.787 |
0.501 |
0.611 |
11.00 |
inscriptis 1.0 (html to txt) |
0.521 |
0.962 |
0.541 |
0.676 |
2.47 |
justext 2.2.0 (German stoplist) |
0.849 |
0.529 |
0.719 |
0.652 |
6.37 |
newspaper 0.2.8 |
0.923 |
0.591 |
0.772 |
0.721 |
14.80 |
goose3 3.1.6 |
0.957 |
0.640 |
0.807 |
0.767 |
21.54 |
boilerpy3 1.0.2 (article mode) |
0.841 |
0.734 |
0.799 |
0.784 |
5.65 |
dragnet 2.0.4 |
0.909 |
0.722 |
0.825 |
0.804 |
3.64 |
readability-lxml 0.7.1 |
0.928 |
0.743 |
0.844 |
0.826 |
6.59 |
news-please 1.4.25 |
0.926 |
0.747 |
0.844 |
0.827 |
70.81 |
trafilatura 0.3.1 (rule-based) |
0.901 |
0.831 |
0.871 |
0.865 |
5.43 |
trafilatura 0.3.1 (+ justext) |
0.897 |
0.868 |
0.884 |
0.882 |
6.97 |
trafilatura 0.4 |
0.914 |
0.869 |
0.894 |
0.891 |
4.87 |
trafilatura 0.4 (+ fallback) |
0.925 |
0.904 |
0.916 |
0.914 |
9.94 |
Older results (2020-01-29)#
100 documents, 266 text and 294 boilerplate segments |
|||||
---|---|---|---|---|---|
Python Package |
Precision |
Recall |
Accuracy |
F-Score |
Time |
raw HTML |
0.492 |
0.902 |
0.511 |
0.637 |
0 |
inscriptis 1.0 (html to txt) |
0.504 |
0.989 |
0.532 |
0.668 |
0.87 |
justext 2.2.0 (German stoplist) |
0.886 |
0.553 |
0.754 |
0.681 |
2.22 |
goose3 3.1.6 |
0.935 |
0.594 |
0.787 |
0.726 |
7.64 |
newspaper 0.2.8 |
0.920 |
0.609 |
0.789 |
0.733 |
5.34 |
boilerpy3 1.0.2 (default mode) |
0.767 |
0.756 |
0.775 |
0.761 |
1.89 |
dragnet 2.0.4 |
0.904 |
0.673 |
0.811 |
0.772 |
1.25 |
readability-lxml 0.7.1 |
0.894 |
0.699 |
0.818 |
0.785 |
2.34 |
news-please 1.4.25 |
0.900 |
0.714 |
0.827 |
0.797 |
22.99 |
trafilatura 0.3.1 (rule-based) |
0.872 |
0.895 |
0.887 |
0.883 |
1.87 |
trafilatura 0.3.1 (+ justext) |
0.889 |
0.936 |
0.914 |
0.912 |
2.19 |