Steve Canny 4711a8dc26
rfctr(part): remove double-decoration 4 (#3690)
**Summary**
Install new `@apply_metadata()` on TXT.

**Additional Context**
- Both EML and MSG delegate to both HTML and TXT to partition the
message-body, depending on which MIME-part body payload is selected
(`text/plain` or `text/html`). This PR prepares the way to remove
decorators from EML and MSG in the next PR.

---------

Co-authored-by: ryannikolaidis <1208590+ryannikolaidis@users.noreply.github.com>
Co-authored-by: scanny <scanny@users.noreply.github.com>
2024-10-03 16:41:31 +00:00

447 lines
14 KiB
Python

# pyright: reportPrivateUsage=false
from __future__ import annotations
import json
import uuid
from typing import Optional, Type
import pytest
from pytest_mock import MockerFixture
from test_unstructured.unit_utils import assert_round_trips_through_JSON, example_doc_path
from unstructured.chunking.title import chunk_by_title
from unstructured.cleaners.core import group_broken_paragraphs
from unstructured.documents.elements import Address, ListItem, NarrativeText, Title
from unstructured.file_utils.model import FileType
from unstructured.partition.text import partition_text
from unstructured.partition.utils.constants import UNSTRUCTURED_INCLUDE_DEBUG_METADATA
EXPECTED_OUTPUT = [
NarrativeText(text="This is a test document to use for unit tests."),
Address(text="Doylestown, PA 18901"),
Title(text="Important points:"),
ListItem(text="Hamburgers are delicious"),
ListItem(text="Dogs are the best"),
ListItem(text="I love fuzzy blankets"),
]
MIN_MAX_TEXT = """This is a story. This is a story that doesn't matter
because it is just being used as an example. Hi. Hello. Howdy. Hola.
The example is simple and repetitive and long and somewhat boring,
but it serves a purpose. End.""".replace(
"\n",
"",
)
SHORT_PARAGRAPHS = """This is a story.
This is a story that doesn't matter because it is just being used as an example.
Hi.
Hello.
Howdy.
Hola.
The example is simple and repetitive and long and somewhat boring, but it serves a purpose.
End.
"""
@pytest.mark.parametrize(
("filename", "encoding"),
[
("fake-text.txt", "utf-8"),
("fake-text.txt", None),
("fake-text-utf-16-be.txt", "utf-16-be"),
],
)
def test_partition_text_from_filename(filename: str, encoding: Optional[str]):
elements = partition_text(example_doc_path(filename), encoding=encoding)
assert len(elements) > 0
assert elements == EXPECTED_OUTPUT
for element in elements:
assert element.metadata.filename == filename
if UNSTRUCTURED_INCLUDE_DEBUG_METADATA:
assert {element.metadata.detection_origin for element in elements} == {"text"}
def test_partition_text_from_filename_with_metadata_filename():
elements = partition_text(
example_doc_path("fake-text.txt"), encoding="utf-8", metadata_filename="test"
)
assert elements == EXPECTED_OUTPUT
for element in elements:
assert element.metadata.filename == "test"
@pytest.mark.parametrize(
"filename",
["fake-text-utf-16.txt", "fake-text-utf-16-le.txt", "fake-text-utf-32.txt"],
)
def test_partition_text_from_filename_default_encoding(filename: str):
elements = partition_text(example_doc_path(filename))
assert len(elements) > 0
assert elements == EXPECTED_OUTPUT
for element in elements:
assert element.metadata.filename == filename
@pytest.mark.parametrize(
("filename", "encoding", "error"),
[
("fake-text.txt", "utf-16", UnicodeDecodeError),
("fake-text-utf-16-be.txt", "utf-16", UnicodeError),
],
)
def test_partition_text_from_filename_raises_econding_error(
filename: str,
encoding: Optional[str],
error: Type[BaseException],
):
with pytest.raises(error):
filename = example_doc_path(filename)
partition_text(filename=filename, encoding=encoding)
def test_partition_text_from_file():
with open(example_doc_path("fake-text.txt"), "rb") as f:
elements = partition_text(file=f)
assert len(elements) > 0
assert elements == EXPECTED_OUTPUT
for element in elements:
assert element.metadata.filename is None
def test_partition_text_from_file_with_metadata_filename():
filename = example_doc_path("fake-text.txt")
with open(filename, "rb") as f:
elements = partition_text(file=f, metadata_filename="test")
assert len(elements) > 0
assert elements == EXPECTED_OUTPUT
for element in elements:
assert element.metadata.filename == "test"
@pytest.mark.parametrize(
"filename",
["fake-text-utf-16.txt", "fake-text-utf-16-le.txt", "fake-text-utf-32.txt"],
)
def test_partition_text_from_file_default_encoding(filename: str):
with open(example_doc_path(filename), "rb") as f:
elements = partition_text(file=f)
assert len(elements) > 0
assert elements == EXPECTED_OUTPUT
for element in elements:
assert element.metadata.filename is None
def test_partition_text_from_bytes_file():
with open(example_doc_path("fake-text.txt"), "rb") as f:
elements = partition_text(file=f)
assert len(elements) > 0
assert elements == EXPECTED_OUTPUT
for element in elements:
assert element.metadata.filename is None
@pytest.mark.parametrize(
"filename",
["fake-text-utf-16.txt", "fake-text-utf-16-le.txt", "fake-text-utf-32.txt"],
)
def test_partition_text_from_bytes_file_default_encoding(filename: str):
with open(example_doc_path(filename), "rb") as f:
elements = partition_text(file=f)
assert len(elements) > 0
assert elements == EXPECTED_OUTPUT
for element in elements:
assert element.metadata.filename is None
def test_text_partition_element_metadata_user_provided_languages():
elements = partition_text(
example_doc_path("book-war-and-peace-1p.txt"), strategy="fast", languages=["en"]
)
assert elements[0].metadata.languages == ["eng"]
def test_partition_text_from_text():
with open(example_doc_path("fake-text.txt")) as f:
text = f.read()
elements = partition_text(text=text)
assert len(elements) > 0
assert elements == EXPECTED_OUTPUT
for element in elements:
assert element.metadata.filename is None
def test_partition_text_from_text_works_with_empty_string():
assert partition_text(text="") == []
def test_partition_text_raises_with_none_specified():
with pytest.raises(ValueError):
partition_text()
def test_partition_text_raises_with_too_many_specified():
filename = example_doc_path("fake-text.txt")
with open(filename) as f:
text = f.read()
with pytest.raises(ValueError):
partition_text(filename=filename, text=text)
def test_partition_text_captures_everything_even_with_linebreaks():
text = """
VERY IMPORTANT MEMO
DOYLESTOWN, PA 18901
"""
elements = partition_text(text=text)
assert elements == [
Title(text="VERY IMPORTANT MEMO"),
Address(text="DOYLESTOWN, PA 18901"),
]
for element in elements:
assert element.metadata.filename is None
def test_partition_text_groups_broken_paragraphs():
text = (
"The big brown fox\n"
"was walking down the lane.\n"
"\n"
"At the end of the lane,\n"
"the fox met a bear."
)
elements = partition_text(text=text, paragraph_grouper=group_broken_paragraphs)
assert elements == [
NarrativeText(text="The big brown fox was walking down the lane."),
NarrativeText(text="At the end of the lane, the fox met a bear."),
]
for element in elements:
assert element.metadata.filename is None
def test_partition_text_splits_long_text():
elements = partition_text(example_doc_path("norwich-city.txt"))
assert len(elements) > 0
assert elements[0].text.startswith("Iwan Roberts")
assert elements[-1].text.endswith("External links")
def test_partition_text_doesnt_get_page_breaks():
text = "--------------------"
elements = partition_text(text=text)
assert len(elements) == 1
assert elements[0].text == text
assert not isinstance(elements[0], ListItem)
# -- .metadata.filename --------------------------------------------------------------------------
def test_partition_text_from_filename_gets_filename_metadata_from_file_path():
elements = partition_text(example_doc_path("fake-text.txt"))
assert all(e.metadata.filename == "fake-text.txt" for e in elements)
assert all(e.metadata.file_directory == example_doc_path("") for e in elements)
def test_partition_text_from_file_gets_filename_metadata_None():
with open(example_doc_path("fake-text.txt"), "rb") as f:
elements = partition_text(file=f)
assert all(e.metadata.filename is None for e in elements)
assert all(e.metadata.file_directory is None for e in elements)
def test_partition_text_from_filename_prefers_metadata_filename():
elements = partition_text(example_doc_path("fake-text.txt"), metadata_filename="a/b/c.txt")
assert all(e.metadata.filename == "c.txt" for e in elements)
assert all(e.metadata.file_directory == "a/b" for e in elements)
def test_partition_text_from_file_prefers_metadata_filename():
with open(example_doc_path("fake-text.txt"), "rb") as f:
elements = partition_text(file=f, metadata_filename="d/e/f.txt")
assert all(e.metadata.filename == "f.txt" for e in elements)
assert all(e.metadata.file_directory == "d/e" for e in elements)
# -- .metadata.filetype --------------------------------------------------------------------------
def test_partition_text_gets_the_TXT_MIME_type_in_metadata_filetype():
TXT_MIME_TYPE = "text/plain"
elements = partition_text(example_doc_path("fake-text.txt"))
assert all(e.metadata.filetype == TXT_MIME_TYPE for e in elements), (
f"Expected all elements to have '{TXT_MIME_TYPE}' as their filetype, but got:"
f" {repr(elements[0].metadata.filetype)}"
)
def test_partition_text_prefers_metadata_file_type():
elements = partition_text(example_doc_path("README.md"), metadata_file_type=FileType.MD)
assert all(e.metadata.filetype == "text/markdown" for e in elements), (
f"Expected all elements to have 'text/markdown' as their filetype, but got:"
f" {repr(elements[0].metadata.filetype)}"
)
# -- .metadata.last_modified ---------------------------------------------------------------------
def test_partition_text_from_file_path_gets_last_modified_from_filesystem(mocker: MockerFixture):
filesystem_last_modified = "2029-07-05T09:24:28"
mocker.patch(
"unstructured.partition.text.get_last_modified_date", return_value=filesystem_last_modified
)
elements = partition_text(example_doc_path("fake-text.txt"))
assert all(e.metadata.last_modified == filesystem_last_modified for e in elements)
def test_partition_text_from_file_gets_last_modified_None():
with open(example_doc_path("fake-text.txt"), "rb") as f:
elements = partition_text(file=f)
assert all(e.metadata.last_modified is None for e in elements)
def test_partition_text_from_text_gets_last_modified_None():
with open(example_doc_path("fake-text.txt")) as f:
text = f.read()
elements = partition_text(text=text)
assert all(e.metadata.last_modified is None for e in elements)
def test_partition_text_from_file_path_prefers_metadata_last_modified(mocker: MockerFixture):
filesystem_last_modified = "2029-07-05T09:24:28"
metadata_last_modified = "2020-07-05T09:24:28"
mocker.patch(
"unstructured.partition.text.get_last_modified_date", return_value=filesystem_last_modified
)
elements = partition_text(
example_doc_path("fake-text.txt"), metadata_last_modified=metadata_last_modified
)
assert all(e.metadata.last_modified == metadata_last_modified for e in elements)
def test_partition_text_from_file_prefers_metadata_last_modified():
metadata_last_modified = "2020-07-05T09:24:28"
with open(example_doc_path("fake-text.txt"), "rb") as f:
elements = partition_text(file=f, metadata_last_modified=metadata_last_modified)
assert all(e.metadata.last_modified == metadata_last_modified for e in elements)
def test_partition_text_from_text_prefers_metadata_last_modified():
metadata_last_modified = "2020-07-05T09:24:28"
with open(example_doc_path("fake-text.txt")) as f:
text = f.read()
elements = partition_text(text=text, metadata_last_modified=metadata_last_modified)
assert all(e.metadata.last_modified == metadata_last_modified for e in elements)
# ------------------------------------------------------------------------------------------------
def test_Text_element_assigns_id_hashes_that_are_unique_and_deterministic():
ids = [element.id for element in partition_text(text="hello\nhello\nhello")]
assert ids == [
"8657c0ec31a4cfc822f6cd4a5684cafd",
"72aefb4a12be063ad160931fdb380163",
"ba8c1a216ca585aecdd365a72e6124f1",
]
def test_Text_element_assings_UUID_when_unique_element_ids_is_True():
elements = partition_text(text="hello\nhello\nhello", unique_element_ids=True)
for element in elements:
assert uuid.UUID(element.id, version=4)
# Test that the element is JSON serializable. This should run without an error
json.dumps(element.to_dict())
@pytest.mark.parametrize(
("file_name", "encoding"),
[
("fake-text.txt", "utf-8"),
("fake-text.txt", None),
("fake-text-utf-16-be.txt", "utf-16-be"),
],
)
def test_partition_text_with_json(file_name: str, encoding: str | None):
elements = partition_text(example_doc_path(file_name), encoding=encoding)
assert_round_trips_through_JSON(elements)
def test_add_chunking_strategy_on_partition_text():
filename = example_doc_path("book-war-and-peace-1p.txt")
elements = partition_text(filename=filename)
chunk_elements = partition_text(filename, chunking_strategy="by_title")
chunks = chunk_by_title(elements)
assert chunk_elements != elements
assert chunk_elements == chunks
def test_partition_text_element_metadata_has_languages():
elements = partition_text(example_doc_path("norwich-city.txt"))
assert elements[0].metadata.languages == ["eng"]
def test_partition_text_respects_detect_language_per_element():
elements = partition_text(
example_doc_path("language-docs/eng_spa_mult.txt"), detect_language_per_element=True
)
langs = [element.metadata.languages for element in elements]
assert langs == [["eng"], ["spa", "eng"], ["eng"], ["eng"], ["spa"]]
def test_partition_text_respects_languages_arg():
elements = partition_text(example_doc_path("norwich-city.txt"), languages=["deu"])
assert elements[0].metadata.languages == ["deu"]
def test_partition_text_element_metadata_raises_TypeError():
with pytest.raises(TypeError):
partition_text(example_doc_path("norwich-city.txt"), languages="eng")
def test_partition_text_detects_more_than_3_languages():
elements = partition_text(
example_doc_path("language-docs/UDHR_first_article_all.txt"),
detect_language_per_element=True,
)
langs = [e.metadata.languages[0] for e in elements if e.metadata.languages]
assert len(langs) > 10