diff --git a/CHANGELOG.md b/CHANGELOG.md index a642da044a..f975f390dc 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,7 @@ ### Enhancements +* **Adds XLSX document level language detection** Enhancing on top of language detection functionality in previous release, we now support language detection within `.xlsx` file type at Element level. * **bump `unstructured-inference` to `0.6.6`** The updated version of `unstructured-inference` makes table extraction in `hi_res` mode configurable to fine tune table extraction performance; it also improves element detection by adding a deduplication post processing step in the `hi_res` partitioning of pdfs and images. * **Detect text in HTML Heading Tags as Titles** This will increase the accuracy of hierarchies in HTML documents and provide more accurate element categorization. If text is in an HTML heading tag and is not a list item, address, or narrative text, categorize it as a title. * **Update python-based docs** Refactor docs to use the actual unstructured code rather than using the subprocess library to run the cli command itself. @@ -10,7 +11,7 @@ * **Expose endpoint url for s3 connectors** By allowing for the endpoint url to be explicitly overwritten, this allows for any non-AWS data providers supporting the s3 protocol to be supported (i.e. minio). * **change default `hi_res` model for pdf/image partition to `yolox`** Now partitioning pdf/image using `hi_res` strategy utilizes `yolox_quantized` model isntead of `detectron2_onnx` model. This new default model has better recall for tables and produces more detailed categories for elements. -### Features +* **XLSX can now reads subtables within one sheet** Problem: Many .xlsx files are not created to be read as one full table per sheet. There are subtables, text and header along with more informations to extract from each sheet. Feature: This `partition_xlsx` now can reads subtable(s) within one .xlsx sheet, along with extracting other title and narrative texts. Importance: This enhance the power of .xlsx reading to not only one table per sheet, allowing user to capture more data tables from the file, if exists. ### Fixes diff --git a/docs/requirements.txt b/docs/requirements.txt index 99386eb5a7..2373ecfb3e 100644 --- a/docs/requirements.txt +++ b/docs/requirements.txt @@ -18,7 +18,7 @@ certifi==2023.7.22 # -c requirements/constraints.in # -r requirements/build.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests @@ -54,7 +54,7 @@ mdurl==0.1.2 # via markdown-it-py myst-parser==2.0.0 # via -r requirements/build.in -packaging==23.1 +packaging==23.2 # via # -c requirements/base.txt # sphinx diff --git a/example-docs/2023-half-year-analyses-by-segment.xlsx b/example-docs/2023-half-year-analyses-by-segment.xlsx new file mode 100644 index 0000000000..d0ce5e673c Binary files /dev/null and b/example-docs/2023-half-year-analyses-by-segment.xlsx differ diff --git a/example-docs/vodafone.xlsx b/example-docs/vodafone.xlsx new file mode 100644 index 0000000000..4467a9301d Binary files /dev/null and b/example-docs/vodafone.xlsx differ diff --git a/requirements/base.txt b/requirements/base.txt index cfc1b241da..3679dd89b2 100644 --- a/requirements/base.txt +++ b/requirements/base.txt @@ -12,7 +12,7 @@ certifi==2023.7.22 # requests chardet==5.2.0 # via -r requirements/base.in -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via requests click==8.1.7 # via nltk @@ -40,7 +40,7 @@ numpy==1.24.4 # via # -c requirements/constraints.in # -r requirements/base.in -packaging==23.1 +packaging==23.2 # via marshmallow python-iso639==2023.6.15 # via -r requirements/base.in diff --git a/requirements/build.txt b/requirements/build.txt index 99386eb5a7..2373ecfb3e 100644 --- a/requirements/build.txt +++ b/requirements/build.txt @@ -18,7 +18,7 @@ certifi==2023.7.22 # -c requirements/constraints.in # -r requirements/build.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests @@ -54,7 +54,7 @@ mdurl==0.1.2 # via markdown-it-py myst-parser==2.0.0 # via -r requirements/build.in -packaging==23.1 +packaging==23.2 # via # -c requirements/base.txt # sphinx diff --git a/requirements/dev.txt b/requirements/dev.txt index b90b4776d6..e83859d7dd 100644 --- a/requirements/dev.txt +++ b/requirements/dev.txt @@ -13,21 +13,18 @@ appnope==0.1.3 # ipykernel # ipython argon2-cffi==23.1.0 - # via jupyter-server + # via + # jupyter-server + # nbclassic + # notebook argon2-cffi-bindings==21.2.0 # via argon2-cffi -arrow==1.2.3 - # via isoduration asttokens==2.4.0 # via stack-data -async-lru==2.0.4 - # via jupyterlab attrs==23.1.0 # via # jsonschema # referencing -babel==2.12.1 - # via jupyterlab-server backcall==0.2.0 # via ipython beautifulsoup4==4.12.2 @@ -36,23 +33,10 @@ beautifulsoup4==4.12.2 # nbconvert bleach==6.0.0 # via nbconvert -build==1.0.3 - # via pip-tools -certifi==2023.7.22 - # via - # -c requirements/base.txt - # -c requirements/constraints.in - # -c requirements/test.txt - # requests cffi==1.16.0 # via argon2-cffi-bindings cfgv==3.4.0 # via pre-commit -charset-normalizer==3.2.0 - # via - # -c requirements/base.txt - # -c requirements/test.txt - # requests click==8.1.7 # via # -c requirements/base.txt @@ -70,47 +54,40 @@ defusedxml==0.7.1 # via nbconvert distlib==0.3.7 # via virtualenv +entrypoints==0.4 + # via + # jupyter-client + # nbconvert exceptiongroup==1.1.3 # via # -c requirements/test.txt # anyio -executing==1.2.0 +executing==2.0.0 # via stack-data -fastjsonschema==2.18.0 +fastjsonschema==2.18.1 # via nbformat filelock==3.12.4 # via virtualenv -fqdn==1.5.1 - # via jsonschema -identify==2.5.29 +identify==2.5.30 # via pre-commit idna==3.4 # via # -c requirements/base.txt # -c requirements/test.txt # anyio - # jsonschema - # requests -importlib-metadata==6.8.0 - # via - # build - # jupyter-client - # jupyter-lsp - # jupyterlab - # jupyterlab-server - # nbconvert importlib-resources==6.1.0 # via # jsonschema # jsonschema-specifications - # jupyterlab -ipykernel==6.25.2 + # notebook +ipykernel==6.11.0 # via # jupyter # jupyter-console - # jupyterlab + # nbclassic + # notebook # qtconsole -ipython==8.12.2 +ipython==8.12.3 # via # -c requirements/constraints.in # -r requirements/dev.in @@ -118,74 +95,55 @@ ipython==8.12.2 # ipywidgets # jupyter-console ipython-genutils==0.2.0 - # via qtconsole + # via + # jupyter-server + # nbclassic + # notebook + # qtconsole ipywidgets==8.1.1 # via jupyter -isoduration==20.11.0 - # via jsonschema jedi==0.19.0 # via ipython jinja2==3.1.2 # via # jupyter-server - # jupyterlab - # jupyterlab-server + # nbclassic # nbconvert -json5==0.9.14 - # via jupyterlab-server -jsonpointer==2.4 - # via jsonschema -jsonschema[format-nongpl]==4.19.1 - # via - # jupyter-events - # jupyterlab-server - # nbformat + # notebook +jsonschema==4.19.1 + # via nbformat jsonschema-specifications==2023.7.1 # via jsonschema jupyter==1.0.0 # via -r requirements/dev.in -jupyter-client==8.3.1 +jupyter-client==7.4.9 # via # ipykernel # jupyter-console # jupyter-server + # nbclassic # nbclient + # notebook # qtconsole -jupyter-console==6.6.3 +jupyter-console==6.4.4 # via jupyter jupyter-core==5.3.2 # via # -c requirements/constraints.in # ipykernel # jupyter-client - # jupyter-console # jupyter-server - # jupyterlab - # nbclient + # nbclassic # nbconvert # nbformat + # notebook # qtconsole -jupyter-events==0.7.0 - # via jupyter-server -jupyter-lsp==2.2.0 - # via jupyterlab -jupyter-server==2.7.3 +jupyter-server==1.13.1 # via - # jupyter-lsp - # jupyterlab - # jupyterlab-server - # notebook + # nbclassic # notebook-shim -jupyter-server-terminals==0.4.4 - # via jupyter-server -jupyterlab==4.0.6 - # via notebook jupyterlab-pygments==0.2.2 # via nbconvert -jupyterlab-server==2.25.0 - # via - # jupyterlab - # notebook jupyterlab-widgets==3.0.9 # via ipywidgets markupsafe==2.1.3 @@ -196,52 +154,53 @@ matplotlib-inline==0.1.6 # via # ipykernel # ipython -mistune==3.0.1 +mistune==0.8.4 # via nbconvert -nbclient==0.8.0 +nbclassic==1.0.0 + # via notebook +nbclient==0.5.13 # via nbconvert -nbconvert==7.8.0 +nbconvert==6.4.5 # via # jupyter # jupyter-server + # nbclassic + # notebook nbformat==5.9.2 # via # jupyter-server + # nbclassic # nbclient # nbconvert + # notebook nest-asyncio==1.5.8 - # via ipykernel + # via + # ipykernel + # jupyter-client + # nbclassic + # nbclient + # notebook nodeenv==1.8.0 # via pre-commit -notebook==7.0.4 +notebook==6.5.6 # via jupyter notebook-shim==0.2.3 # via - # jupyterlab + # nbclassic # notebook -overrides==7.4.0 - # via jupyter-server -packaging==23.1 - # via - # -c requirements/base.txt - # -c requirements/test.txt - # build - # ipykernel - # jupyter-server - # jupyterlab - # jupyterlab-server - # nbconvert - # qtconsole - # qtpy pandocfilters==1.5.0 # via nbconvert parso==0.8.3 # via jedi +pep517==0.13.0 + # via + # build + # pip-tools pexpect==4.8.0 # via ipython pickleshare==0.7.5 # via ipython -pip-tools==7.3.0 +pip-tools==6.6.2 # via -r requirements/dev.in pkgutil-resolve-name==1.3.10 # via jsonschema @@ -253,7 +212,10 @@ platformdirs==3.10.0 pre-commit==3.4.0 # via -r requirements/dev.in prometheus-client==0.17.1 - # via jupyter-server + # via + # jupyter-server + # nbclassic + # notebook prompt-toolkit==3.0.39 # via # ipython @@ -274,57 +236,38 @@ pygments==2.16.1 # jupyter-console # nbconvert # qtconsole -pyproject-hooks==1.0.0 - # via build python-dateutil==2.8.2 # via # -c requirements/test.txt - # arrow # jupyter-client -python-json-logger==2.0.7 - # via jupyter-events -pytz==2023.3.post1 - # via babel pyyaml==6.0.1 # via # -c requirements/test.txt - # jupyter-events # pre-commit -pyzmq==25.1.1 +pyzmq==24.0.1 # via - # ipykernel # jupyter-client - # jupyter-console # jupyter-server + # nbclassic + # notebook # qtconsole -qtconsole==5.4.4 +qtconsole==5.2.2 # via jupyter -qtpy==2.4.0 +qtpy==1.11.3 # via qtconsole referencing==0.30.2 # via # jsonschema # jsonschema-specifications - # jupyter-events -requests==2.31.0 - # via - # -c requirements/base.txt - # -c requirements/test.txt - # jupyterlab-server -rfc3339-validator==0.1.4 - # via - # jsonschema - # jupyter-events -rfc3986-validator==0.1.1 - # via - # jsonschema - # jupyter-events rpds-py==0.10.3 # via # jsonschema # referencing send2trash==1.8.2 - # via jupyter-server + # via + # jupyter-server + # nbclassic + # notebook six==1.16.0 # via # -c requirements/base.txt @@ -332,34 +275,32 @@ six==1.16.0 # asttokens # bleach # python-dateutil - # rfc3339-validator sniffio==1.3.0 # via anyio soupsieve==2.5 # via # -c requirements/base.txt # beautifulsoup4 -stack-data==0.6.2 +stack-data==0.6.3 # via ipython terminado==0.17.1 # via # jupyter-server - # jupyter-server-terminals -tinycss2==1.2.1 + # nbclassic + # notebook +testpath==0.6.0 # via nbconvert tomli==2.0.1 # via # -c requirements/test.txt - # build - # jupyterlab + # pep517 # pip-tools - # pyproject-hooks tornado==6.3.3 # via # ipykernel # jupyter-client # jupyter-server - # jupyterlab + # nbclassic # notebook # terminado traitlets==5.10.1 @@ -369,40 +310,26 @@ traitlets==5.10.1 # ipython # ipywidgets # jupyter-client - # jupyter-console # jupyter-core - # jupyter-events # jupyter-server - # jupyterlab # matplotlib-inline + # nbclassic # nbclient # nbconvert # nbformat + # notebook # qtconsole typing-extensions==4.8.0 # via # -c requirements/base.txt # -c requirements/test.txt - # async-lru # ipython -uri-template==1.3.0 - # via jsonschema -urllib3==1.26.16 - # via - # -c requirements/base.txt - # -c requirements/constraints.in - # -c requirements/test.txt - # requests virtualenv==20.24.5 # via pre-commit -wcwidth==0.2.7 +wcwidth==0.2.8 # via prompt-toolkit -webcolors==1.13 - # via jsonschema webencodings==0.5.1 - # via - # bleach - # tinycss2 + # via bleach websocket-client==1.6.3 # via jupyter-server wheel==0.41.2 @@ -412,9 +339,7 @@ wheel==0.41.2 widgetsnbextension==4.0.9 # via ipywidgets zipp==3.17.0 - # via - # importlib-metadata - # importlib-resources + # via importlib-resources # The following packages are considered to be unsafe in a requirements file: # pip diff --git a/requirements/extra-paddleocr.txt b/requirements/extra-paddleocr.txt index 1f028530d2..7a3fc605d0 100644 --- a/requirements/extra-paddleocr.txt +++ b/requirements/extra-paddleocr.txt @@ -19,7 +19,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests @@ -39,13 +39,13 @@ cython==3.0.2 # via unstructured-paddleocr et-xmlfile==1.1.0 # via openpyxl -flask==2.3.3 +flask==3.0.0 # via # flask-babel # visualdl flask-babel==3.1.0 # via visualdl -fonttools==4.42.1 +fonttools==4.43.0 # via matplotlib future==0.18.3 # via bce-python-sdk @@ -122,7 +122,7 @@ opencv-python==4.8.0.76 # unstructured-paddleocr openpyxl==3.1.2 # via unstructured-paddleocr -packaging==23.1 +packaging==23.2 # via # -c requirements/base.txt # matplotlib @@ -218,7 +218,7 @@ urllib3==1.26.16 # requests visualdl==2.5.3 # via unstructured-paddleocr -werkzeug==2.3.7 +werkzeug==3.0.0 # via flask zipp==3.17.0 # via diff --git a/requirements/extra-pdf-image.txt b/requirements/extra-pdf-image.txt index 679ffef54c..140eaa0a24 100644 --- a/requirements/extra-pdf-image.txt +++ b/requirements/extra-pdf-image.txt @@ -13,7 +13,7 @@ certifi==2023.7.22 # requests cffi==1.16.0 # via cryptography -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # pdfminer-six @@ -35,7 +35,7 @@ filelock==3.12.4 # transformers flatbuffers==23.5.26 # via onnxruntime -fonttools==4.42.1 +fonttools==4.43.0 # via matplotlib fsspec==2023.9.1 # via @@ -98,7 +98,7 @@ opencv-python==4.8.0.76 # -c requirements/constraints.in # layoutparser # unstructured-inference -packaging==23.1 +packaging==23.2 # via # -c requirements/base.txt # huggingface-hub diff --git a/requirements/extra-pptx.txt b/requirements/extra-pptx.txt index 9e30fd6f95..a0d861f1e5 100644 --- a/requirements/extra-pptx.txt +++ b/requirements/extra-pptx.txt @@ -10,5 +10,5 @@ pillow==10.0.1 # via python-pptx python-pptx==0.6.21 # via -r requirements/extra-pptx.in -xlsxwriter==3.1.5 +xlsxwriter==3.1.6 # via python-pptx diff --git a/requirements/huggingface.txt b/requirements/huggingface.txt index 00ba71293a..2fc6f0efb9 100644 --- a/requirements/huggingface.txt +++ b/requirements/huggingface.txt @@ -9,7 +9,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests @@ -53,7 +53,7 @@ numpy==1.24.4 # -c requirements/base.txt # -c requirements/constraints.in # transformers -packaging==23.1 +packaging==23.2 # via # -c requirements/base.txt # huggingface-hub diff --git a/requirements/ingest-airtable.txt b/requirements/ingest-airtable.txt index db7e92a6a1..26744992b7 100644 --- a/requirements/ingest-airtable.txt +++ b/requirements/ingest-airtable.txt @@ -9,7 +9,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/ingest-azure-cognitive-search.txt b/requirements/ingest-azure-cognitive-search.txt index 19e918acb6..763e3a14c1 100644 --- a/requirements/ingest-azure-cognitive-search.txt +++ b/requirements/ingest-azure-cognitive-search.txt @@ -18,7 +18,7 @@ certifi==2023.7.22 # -c requirements/constraints.in # msrest # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/ingest-azure.txt b/requirements/ingest-azure.txt index e682d29422..55370e769f 100644 --- a/requirements/ingest-azure.txt +++ b/requirements/ingest-azure.txt @@ -34,7 +34,7 @@ cffi==1.16.0 # via # azure-datalake-store # cryptography -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # aiohttp @@ -61,7 +61,7 @@ idna==3.4 # yarl isodate==0.6.1 # via azure-storage-blob -msal==1.24.0 +msal==1.24.1 # via # azure-datalake-store # azure-identity diff --git a/requirements/ingest-box.txt b/requirements/ingest-box.txt index 79268b6b3d..5d61bfc721 100644 --- a/requirements/ingest-box.txt +++ b/requirements/ingest-box.txt @@ -17,7 +17,7 @@ certifi==2023.7.22 # requests cffi==1.16.0 # via cryptography -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/ingest-confluence.txt b/requirements/ingest-confluence.txt index cbcf462cf4..0a36bb3cfa 100644 --- a/requirements/ingest-confluence.txt +++ b/requirements/ingest-confluence.txt @@ -11,7 +11,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/ingest-discord.txt b/requirements/ingest-discord.txt index 72d0250723..0b0800082e 100644 --- a/requirements/ingest-discord.txt +++ b/requirements/ingest-discord.txt @@ -12,7 +12,7 @@ async-timeout==4.0.3 # via aiohttp attrs==23.1.0 # via aiohttp -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # aiohttp diff --git a/requirements/ingest-dropbox.txt b/requirements/ingest-dropbox.txt index f9b094be42..56c77ff37e 100644 --- a/requirements/ingest-dropbox.txt +++ b/requirements/ingest-dropbox.txt @@ -9,7 +9,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/ingest-gcs.txt b/requirements/ingest-gcs.txt index 4f6d048137..463dcbddff 100644 --- a/requirements/ingest-gcs.txt +++ b/requirements/ingest-gcs.txt @@ -25,7 +25,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # aiohttp diff --git a/requirements/ingest-github.txt b/requirements/ingest-github.txt index ad5ac2a7a0..1c649274c5 100644 --- a/requirements/ingest-github.txt +++ b/requirements/ingest-github.txt @@ -13,7 +13,7 @@ cffi==1.16.0 # via # cryptography # pynacl -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests @@ -27,20 +27,31 @@ idna==3.4 # requests pycparser==2.21 # via cffi -pygithub==1.59.1 +pygithub==2.1.1 # via -r requirements/ingest-github.in pyjwt[crypto]==2.8.0 # via pygithub pynacl==1.5.0 # via pygithub +python-dateutil==2.8.2 + # via pygithub requests==2.31.0 # via # -c requirements/base.txt # pygithub +six==1.16.0 + # via + # -c requirements/base.txt + # python-dateutil +typing-extensions==4.8.0 + # via + # -c requirements/base.txt + # pygithub urllib3==1.26.16 # via # -c requirements/base.txt # -c requirements/constraints.in + # pygithub # requests wrapt==1.15.0 # via deprecated diff --git a/requirements/ingest-gitlab.txt b/requirements/ingest-gitlab.txt index dbff64042c..4d45eeda5c 100644 --- a/requirements/ingest-gitlab.txt +++ b/requirements/ingest-gitlab.txt @@ -9,7 +9,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/ingest-google-drive.txt b/requirements/ingest-google-drive.txt index 9f90bcc9ca..ace1ff7fa2 100644 --- a/requirements/ingest-google-drive.txt +++ b/requirements/ingest-google-drive.txt @@ -11,7 +11,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/ingest-jira.txt b/requirements/ingest-jira.txt index adb8173b2c..e53d3dc493 100644 --- a/requirements/ingest-jira.txt +++ b/requirements/ingest-jira.txt @@ -11,7 +11,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/ingest-onedrive.txt b/requirements/ingest-onedrive.txt index 2d9627f1d4..99e838c70b 100644 --- a/requirements/ingest-onedrive.txt +++ b/requirements/ingest-onedrive.txt @@ -17,7 +17,7 @@ certifi==2023.7.22 # requests cffi==1.16.0 # via cryptography -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests @@ -29,7 +29,7 @@ idna==3.4 # via # -c requirements/base.txt # requests -msal==1.24.0 +msal==1.24.1 # via # -r requirements/ingest-onedrive.in # office365-rest-python-client diff --git a/requirements/ingest-openai.txt b/requirements/ingest-openai.txt index d7846c0a08..d0562f7058 100644 --- a/requirements/ingest-openai.txt +++ b/requirements/ingest-openai.txt @@ -25,7 +25,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # aiohttp @@ -50,7 +50,7 @@ jsonpatch==1.33 # via langchain jsonpointer==2.4 # via jsonpatch -langchain==0.0.304 +langchain==0.0.305 # via -r requirements/ingest-openai.in langsmith==0.0.41 # via langchain @@ -76,7 +76,7 @@ numpy==1.24.4 # numexpr openai==0.28.1 # via -r requirements/ingest-openai.in -packaging==23.1 +packaging==23.2 # via # -c requirements/base.txt # marshmallow diff --git a/requirements/ingest-outlook.txt b/requirements/ingest-outlook.txt index ccef36d349..9ca3f43a72 100644 --- a/requirements/ingest-outlook.txt +++ b/requirements/ingest-outlook.txt @@ -11,7 +11,7 @@ certifi==2023.7.22 # requests cffi==1.16.0 # via cryptography -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests @@ -23,7 +23,7 @@ idna==3.4 # via # -c requirements/base.txt # requests -msal==1.24.0 +msal==1.24.1 # via # -r requirements/ingest-outlook.in # office365-rest-python-client diff --git a/requirements/ingest-reddit.txt b/requirements/ingest-reddit.txt index 8c0b53b240..7e19fdb9f4 100644 --- a/requirements/ingest-reddit.txt +++ b/requirements/ingest-reddit.txt @@ -9,7 +9,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/ingest-s3.txt b/requirements/ingest-s3.txt index d15a09eba9..b86dfe415b 100644 --- a/requirements/ingest-s3.txt +++ b/requirements/ingest-s3.txt @@ -20,7 +20,7 @@ attrs==23.1.0 # via aiohttp botocore==1.31.17 # via aiobotocore -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # aiohttp diff --git a/requirements/ingest-salesforce.txt b/requirements/ingest-salesforce.txt index a6c31b1014..817921dd71 100644 --- a/requirements/ingest-salesforce.txt +++ b/requirements/ingest-salesforce.txt @@ -13,7 +13,7 @@ certifi==2023.7.22 # requests cffi==1.16.0 # via cryptography -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/ingest-sharepoint.txt b/requirements/ingest-sharepoint.txt index 99d1efbfde..97cae3dd91 100644 --- a/requirements/ingest-sharepoint.txt +++ b/requirements/ingest-sharepoint.txt @@ -11,7 +11,7 @@ certifi==2023.7.22 # requests cffi==1.16.0 # via cryptography -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests @@ -23,7 +23,7 @@ idna==3.4 # via # -c requirements/base.txt # requests -msal==1.24.0 +msal==1.24.1 # via # -r requirements/ingest-sharepoint.in # office365-rest-python-client diff --git a/requirements/ingest-wikipedia.txt b/requirements/ingest-wikipedia.txt index e391f0156e..ec1add403a 100644 --- a/requirements/ingest-wikipedia.txt +++ b/requirements/ingest-wikipedia.txt @@ -13,7 +13,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests diff --git a/requirements/test.txt b/requirements/test.txt index 98d40fd188..b4e48463db 100644 --- a/requirements/test.txt +++ b/requirements/test.txt @@ -13,7 +13,7 @@ certifi==2023.7.22 # -c requirements/base.txt # -c requirements/constraints.in # requests -charset-normalizer==3.2.0 +charset-normalizer==3.3.0 # via # -c requirements/base.txt # requests @@ -32,7 +32,7 @@ flake8==6.1.0 # via -r requirements/test.in freezegun==1.2.2 # via -r requirements/test.in -grpcio==1.58.0 +grpcio==1.59.0 # via -r requirements/test.in idna==3.4 # via @@ -61,7 +61,7 @@ mypy-extensions==1.0.0 # -c requirements/base.txt # black # mypy -packaging==23.1 +packaging==23.2 # via # -c requirements/base.txt # black diff --git a/test_unstructured/partition/test_auto.py b/test_unstructured/partition/test_auto.py index e8404537c1..a141f26ee0 100644 --- a/test_unstructured/partition/test_auto.py +++ b/test_unstructured/partition/test_auto.py @@ -8,7 +8,7 @@ import docx import pytest -from test_unstructured.partition.test_constants import EXPECTED_TABLE, EXPECTED_TEXT +from test_unstructured.partition.test_constants import EXPECTED_TABLE, EXPECTED_TEXT, EXPECTED_TITLE from unstructured.chunking.title import chunk_by_title from unstructured.cleaners.core import clean_extra_whitespace from unstructured.documents.elements import ( @@ -708,36 +708,51 @@ def test_auto_partition_xml_from_file_with_tags(filename="example-docs/factbook. def test_auto_partition_xlsx_from_filename(filename="example-docs/stanley-cups.xlsx"): elements = partition(filename=filename, include_header=False) - assert all(isinstance(element, Table) for element in elements) - assert len(elements) == 2 + assert sum(isinstance(element, Table) for element in elements) == 2 + assert sum(isinstance(element, Title) for element in elements) == 2 + assert len(elements) == 4 - assert clean_extra_whitespace(elements[0].text) == EXPECTED_TEXT - assert elements[0].metadata.text_as_html == EXPECTED_TABLE - assert elements[0].metadata.page_number == 1 - assert elements[0].metadata.filetype == EXPECTED_XLSX_FILETYPE + assert clean_extra_whitespace(elements[0].text) == EXPECTED_TITLE + assert clean_extra_whitespace(elements[1].text) == EXPECTED_TEXT + assert elements[1].metadata.text_as_html == EXPECTED_TABLE + assert elements[1].metadata.page_number == 1 + assert elements[1].metadata.filetype == EXPECTED_XLSX_FILETYPE def test_auto_partition_xlsx_from_file(filename="example-docs/stanley-cups.xlsx"): with open(filename, "rb") as f: elements = partition(file=f, include_header=False) - assert all(isinstance(element, Table) for element in elements) - assert len(elements) == 2 + assert sum(isinstance(element, Table) for element in elements) == 2 + assert sum(isinstance(element, Title) for element in elements) == 2 + assert len(elements) == 4 - assert clean_extra_whitespace(elements[0].text) == EXPECTED_TEXT - assert elements[0].metadata.text_as_html == EXPECTED_TABLE - assert elements[0].metadata.page_number == 1 - assert elements[0].metadata.filetype == EXPECTED_XLSX_FILETYPE + assert clean_extra_whitespace(elements[0].text) == EXPECTED_TITLE + assert clean_extra_whitespace(elements[1].text) == EXPECTED_TEXT + assert elements[1].metadata.text_as_html == EXPECTED_TABLE + assert elements[1].metadata.page_number == 1 + assert elements[1].metadata.filetype == EXPECTED_XLSX_FILETYPE -EXPECTED_XLS_TEXT_LEN = 507 +EXPECTED_XLS_TEXT_LEN = 550 -EXPECTED_XLS_INITIAL_45_CLEAN_TEXT = "MA What C datatypes are 8 bits? (assume i386)" +EXPECTED_XLS_INITIAL_45_CLEAN_TEXT = "MC What is 2+2? 4 correct 3 incorrect MA What" EXPECTED_XLS_TABLE = ( """ + + + + + + + + + + + @@ -814,8 +829,8 @@ def test_auto_partition_xlsx_from_file(filename="example-docs/stanley-cups.xlsx" def test_auto_partition_xls_from_filename(filename="example-docs/tests-example.xls"): elements = partition(filename=filename, include_header=False) - assert all(isinstance(element, Table) for element in elements) - assert len(elements) == 3 + assert sum(isinstance(element, Table) for element in elements) == 2 + assert len(elements) == 18 assert clean_extra_whitespace(elements[0].text)[:45] == EXPECTED_XLS_INITIAL_45_CLEAN_TEXT # NOTE(crag): if the beautifulsoup4 package is installed, some (but not all) additional diff --git a/test_unstructured/partition/test_constants.py b/test_unstructured/partition/test_constants.py index f9f04baad0..3dd978d86a 100644 --- a/test_unstructured/partition/test_constants.py +++ b/test_unstructured/partition/test_constants.py @@ -23,6 +23,7 @@
MCWhat is 2+2?4correct3incorrect
MA What C datatypes are 8 bits? (assume i386)
""" +EXPECTED_TITLE = "Stanley Cups" EXPECTED_TEXT = "Team Location Stanley Cups Blues STL 1 Flyers PHI 2 Maple Leafs TOR 13" diff --git a/test_unstructured/partition/xlsx/test_xlsx.py b/test_unstructured/partition/xlsx/test_xlsx.py index a541fe5566..b3e564bc9d 100644 --- a/test_unstructured/partition/xlsx/test_xlsx.py +++ b/test_unstructured/partition/xlsx/test_xlsx.py @@ -1,6 +1,8 @@ -from test_unstructured.partition.test_constants import EXPECTED_TABLE, EXPECTED_TEXT +import pytest + +from test_unstructured.partition.test_constants import EXPECTED_TABLE, EXPECTED_TEXT, EXPECTED_TITLE from unstructured.cleaners.core import clean_extra_whitespace -from unstructured.documents.elements import Table +from unstructured.documents.elements import Table, Text, Title from unstructured.partition.json import partition_json from unstructured.partition.xlsx import partition_xlsx from unstructured.staging.base import elements_to_json @@ -13,20 +15,21 @@ def test_partition_xlsx_from_filename(filename="example-docs/stanley-cups.xlsx"): elements = partition_xlsx(filename=filename, include_header=False) - assert all(isinstance(element, Table) for element in elements) - assert len(elements) == 2 + assert sum(isinstance(element, Table) for element in elements) == 2 + assert len(elements) == 4 - assert clean_extra_whitespace(elements[0].text) == EXPECTED_TEXT - assert elements[0].metadata.text_as_html == EXPECTED_TABLE - assert elements[0].metadata.page_number == 1 - assert elements[0].metadata.filetype == EXPECTED_FILETYPE - assert elements[0].metadata.page_name == EXCEPTED_PAGE_NAME - assert elements[0].metadata.filename == "stanley-cups.xlsx" + assert clean_extra_whitespace(elements[0].text) == EXPECTED_TITLE + assert clean_extra_whitespace(elements[1].text) == EXPECTED_TEXT + assert elements[1].metadata.text_as_html == EXPECTED_TABLE + assert elements[1].metadata.page_number == 1 + assert elements[1].metadata.filetype == EXPECTED_FILETYPE + assert elements[1].metadata.page_name == EXCEPTED_PAGE_NAME + assert elements[1].metadata.filename == "stanley-cups.xlsx" def test_partition_xlsx_from_filename_with_emoji(filename="example-docs/emoji.xlsx"): elements = partition_xlsx(filename=filename, include_header=False) - assert all(isinstance(element, Table) for element in elements) + assert sum(isinstance(element, Text) for element in elements) == 1 assert len(elements) == 1 assert clean_extra_whitespace(elements[0].text) == "🤠😅" @@ -36,14 +39,16 @@ def test_partition_xlsx_from_filename_with_metadata_filename( ): elements = partition_xlsx(filename=filename, metadata_filename="test", include_header=False) - assert all(isinstance(element, Table) for element in elements) - assert clean_extra_whitespace(elements[0].text) == EXPECTED_TEXT + assert sum(isinstance(element, Table) for element in elements) == 2 + assert sum(isinstance(element, Title) for element in elements) == 2 + assert clean_extra_whitespace(elements[0].text) == EXPECTED_TITLE + assert clean_extra_whitespace(elements[1].text) == EXPECTED_TEXT assert elements[0].metadata.filename == "test" def test_partition_xlsx_from_filename_with_header(filename="example-docs/stanley-cups.xlsx"): elements = partition_xlsx(filename=filename, include_header=True) - assert all(isinstance(element, Table) for element in elements) + assert sum(isinstance(element, Table) for element in elements) == 2 assert len(elements) == 2 assert ( clean_extra_whitespace(elements[0].text) @@ -56,30 +61,31 @@ def test_partition_xlsx_from_file(filename="example-docs/stanley-cups.xlsx"): with open(filename, "rb") as f: elements = partition_xlsx(file=f, include_header=False) - assert all(isinstance(element, Table) for element in elements) - assert len(elements) == 2 - assert clean_extra_whitespace(elements[0].text) == EXPECTED_TEXT - assert elements[0].metadata.text_as_html == EXPECTED_TABLE - assert elements[0].metadata.page_number == 1 - assert elements[0].metadata.filetype == EXPECTED_FILETYPE - assert elements[0].metadata.page_name == EXCEPTED_PAGE_NAME - assert elements[0].metadata.filename is None + assert sum(isinstance(element, Table) for element in elements) == 2 + assert len(elements) == 4 + assert clean_extra_whitespace(elements[0].text) == EXPECTED_TITLE + assert clean_extra_whitespace(elements[1].text) == EXPECTED_TEXT + assert elements[1].metadata.text_as_html == EXPECTED_TABLE + assert elements[1].metadata.page_number == 1 + assert elements[1].metadata.filetype == EXPECTED_FILETYPE + assert elements[1].metadata.page_name == EXCEPTED_PAGE_NAME + assert elements[1].metadata.filename is None def test_partition_xlsx_from_file_with_metadata_filename(filename="example-docs/stanley-cups.xlsx"): with open(filename, "rb") as f: elements = partition_xlsx(file=f, metadata_filename="test", include_header=False) - assert all(isinstance(element, Table) for element in elements) - assert clean_extra_whitespace(elements[0].text) == EXPECTED_TEXT - assert elements[0].metadata.filename == "test" + assert sum(isinstance(element, Table) for element in elements) == 2 + assert clean_extra_whitespace(elements[1].text) == EXPECTED_TEXT + assert elements[1].metadata.filename == "test" def test_partition_xlsx_from_file_with_header(filename="example-docs/stanley-cups.xlsx"): with open(filename, "rb") as f: elements = partition_xlsx(file=f, include_header=True) - assert all(isinstance(element, Table) for element in elements) + assert sum(isinstance(element, Table) for element in elements) == 2 assert len(elements) == 2 assert ( clean_extra_whitespace(elements[0].text) @@ -91,25 +97,27 @@ def test_partition_xlsx_from_file_with_header(filename="example-docs/stanley-cup def test_partition_xlsx_filename_exclude_metadata(filename="example-docs/stanley-cups.xlsx"): elements = partition_xlsx(filename=filename, include_metadata=False, include_header=False) - assert all(isinstance(element, Table) for element in elements) - assert len(elements) == 2 + assert sum(isinstance(element, Table) for element in elements) == 2 + assert len(elements) == 4 - assert clean_extra_whitespace(elements[0].text) == EXPECTED_TEXT - assert elements[0].metadata.text_as_html is None - assert elements[0].metadata.page_number is None - assert elements[0].metadata.filetype is None - assert elements[0].metadata.page_name is None - assert elements[0].metadata.filename is None + assert clean_extra_whitespace(elements[1].text) == EXPECTED_TEXT + assert elements[1].metadata.text_as_html is None + assert elements[1].metadata.page_number is None + assert elements[1].metadata.filetype is None + assert elements[1].metadata.page_name is None + assert elements[1].metadata.filename is None def test_partition_xlsx_from_file_exclude_metadata(filename="example-docs/stanley-cups.xlsx"): with open(filename, "rb") as f: elements = partition_xlsx(file=f, include_metadata=False, include_header=False) - assert all(isinstance(element, Table) for element in elements) - assert len(elements) == 2 + assert sum(isinstance(element, Table) for element in elements) == 2 + assert sum(isinstance(element, Title) for element in elements) == 2 + assert len(elements) == 4 - assert clean_extra_whitespace(elements[0].text) == EXPECTED_TEXT + assert clean_extra_whitespace(elements[0].text) == EXPECTED_TITLE + assert clean_extra_whitespace(elements[1].text) == EXPECTED_TEXT assert elements[0].metadata.text_as_html is None assert elements[0].metadata.page_number is None assert elements[0].metadata.filetype is None @@ -205,3 +213,19 @@ def test_partition_xlsx_with_json(filename="example-docs/stanley-cups.xlsx"): for i in range(len(elements)): assert elements[i] == test_elements[i] + + +@pytest.mark.skip("Needs to fix language detection for table. Currently detected as 'tur'") +def test_partition_xlsx_metadata_language_from_filename(filename="example-docs/stanley-cups.xlsx"): + elements = partition_xlsx(filename=filename, include_header=False) + + assert sum(isinstance(element, Table) for element in elements) == 2 + assert len(elements) == 4 + + assert elements[0].metadata.languages == ["eng"] + + +def test_partition_xlsx_subtables(filename="example-docs/vodafone.xlsx"): + elements = partition_xlsx(filename) + assert sum(isinstance(element, Table) for element in elements) == 3 + assert len(elements) == 6 diff --git a/test_unstructured_ingest/expected-structured-output/Sharepoint/Shared Documents/stanley-cups.xlsx.json b/test_unstructured_ingest/expected-structured-output/Sharepoint/Shared Documents/stanley-cups.xlsx.json index 596ff5e799..99079673a7 100644 --- a/test_unstructured_ingest/expected-structured-output/Sharepoint/Shared Documents/stanley-cups.xlsx.json +++ b/test_unstructured_ingest/expected-structured-output/Sharepoint/Shared Documents/stanley-cups.xlsx.json @@ -1,7 +1,32 @@ [ + { + "type": "Title", + "element_id": "c37e2cb941a2e20a9f728fbea5f9e400", + "metadata": { + "data_source": { + "url": "https://unstructuredio.sharepoint.com/Shared Documents/stanley-cups.xlsx?d=wb9956a338079432191ea609def07394d", + "version": 1, + "record_locator": { + "server_path": "/Shared Documents/stanley-cups.xlsx", + "site_url": "https://unstructuredio.sharepoint.com" + }, + "date_created": "2023-06-16T05:05:05", + "date_modified": "2023-06-16T05:05:05" + }, + "filename": "stanley-cups.xlsx", + "filetype": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", + "languages": [ + "tur" + ], + "page_number": 1, + "page_name": "Stanley Cups", + "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR13
" + }, + "text": "Stanley Cups" + }, { "type": "Table", - "element_id": "3e65b02bec20bb1056bd23a3b4ecd0f6", + "element_id": "c00fc0e5ac303c40f9089791e5e485b1", "metadata": { "data_source": { "url": "https://unstructuredio.sharepoint.com/Shared Documents/stanley-cups.xlsx?d=wb9956a338079432191ea609def07394d", @@ -15,15 +40,43 @@ }, "filename": "stanley-cups.xlsx", "filetype": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", + "languages": [ + "tur" + ], "page_number": 1, "page_name": "Stanley Cups", - "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
Stanley CupsUnnamed: 1Unnamed: 2
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR13
" + "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR13
" + }, + "text": "\n\n\nTeam\nLocation\nStanley Cups\n\n\nBlues\nSTL\n1\n\n\nFlyers\nPHI\n2\n\n\nMaple Leafs\nTOR\n13\n\n\n" + }, + { + "type": "Title", + "element_id": "98656277bdadc9ef7d1a9e1bc969579b", + "metadata": { + "data_source": { + "url": "https://unstructuredio.sharepoint.com/Shared Documents/stanley-cups.xlsx?d=wb9956a338079432191ea609def07394d", + "version": 1, + "record_locator": { + "server_path": "/Shared Documents/stanley-cups.xlsx", + "site_url": "https://unstructuredio.sharepoint.com" + }, + "date_created": "2023-06-16T05:05:05", + "date_modified": "2023-06-16T05:05:05" + }, + "filename": "stanley-cups.xlsx", + "filetype": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", + "languages": [ + "tur" + ], + "page_number": 2, + "page_name": "Stanley Cups Since 67", + "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR0
" }, - "text": "\n\n\nStanley Cups\nUnnamed: 1\nUnnamed: 2\n\n\n\n\nTeam\nLocation\nStanley Cups\n\n\nBlues\nSTL\n1\n\n\nFlyers\nPHI\n2\n\n\nMaple Leafs\nTOR\n13\n\n\n" + "text": "Stanley Cups Since 67" }, { "type": "Table", - "element_id": "0699dddf33814117e04654068f5182f6", + "element_id": "31421b5cd94fedb10dc82738503b4505", "metadata": { "data_source": { "url": "https://unstructuredio.sharepoint.com/Shared Documents/stanley-cups.xlsx?d=wb9956a338079432191ea609def07394d", @@ -37,10 +90,13 @@ }, "filename": "stanley-cups.xlsx", "filetype": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", + "languages": [ + "tur" + ], "page_number": 2, "page_name": "Stanley Cups Since 67", - "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
Stanley Cups Since 67Unnamed: 1Unnamed: 2
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR0
" + "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR0
" }, - "text": "\n\n\nStanley Cups Since 67\nUnnamed: 1\nUnnamed: 2\n\n\n\n\nTeam\nLocation\nStanley Cups\n\n\nBlues\nSTL\n1\n\n\nFlyers\nPHI\n2\n\n\nMaple Leafs\nTOR\n0\n\n\n" + "text": "\n\n\nTeam\nLocation\nStanley Cups\n\n\nBlues\nSTL\n1\n\n\nFlyers\nPHI\n2\n\n\nMaple Leafs\nTOR\n0\n\n\n" } ] \ No newline at end of file diff --git a/test_unstructured_ingest/expected-structured-output/gcs/nested-2/stanley-cups.xlsx.json b/test_unstructured_ingest/expected-structured-output/gcs/nested-2/stanley-cups.xlsx.json index c26b2cda9d..79cebfc45b 100644 --- a/test_unstructured_ingest/expected-structured-output/gcs/nested-2/stanley-cups.xlsx.json +++ b/test_unstructured_ingest/expected-structured-output/gcs/nested-2/stanley-cups.xlsx.json @@ -1,7 +1,31 @@ [ + { + "type": "Title", + "element_id": "c37e2cb941a2e20a9f728fbea5f9e400", + "metadata": { + "data_source": { + "url": "gs://utic-test-ingest-fixtures/nested-2/stanley-cups.xlsx", + "version": "COul9MuE0/8CEAE=", + "record_locator": { + "protocol": "gs", + "remote_file_path": "utic-test-ingest-fixtures/nested-2/stanley-cups.xlsx" + }, + "date_created": "2023-06-20T23:48:24.973000+00:00", + "date_modified": "2023-06-20T23:48:24.973000+00:00" + }, + "filetype": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", + "languages": [ + "tur" + ], + "page_number": 1, + "page_name": "Stanley Cups", + "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR13
" + }, + "text": "Stanley Cups" + }, { "type": "Table", - "element_id": "3e65b02bec20bb1056bd23a3b4ecd0f6", + "element_id": "c00fc0e5ac303c40f9089791e5e485b1", "metadata": { "data_source": { "url": "gs://utic-test-ingest-fixtures/nested-2/stanley-cups.xlsx", @@ -14,15 +38,42 @@ "date_modified": "2023-06-20T23:48:24.973000+00:00" }, "filetype": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", + "languages": [ + "tur" + ], "page_number": 1, "page_name": "Stanley Cups", - "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
Stanley CupsUnnamed: 1Unnamed: 2
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR13
" + "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR13
" + }, + "text": "\n\n\nTeam\nLocation\nStanley Cups\n\n\nBlues\nSTL\n1\n\n\nFlyers\nPHI\n2\n\n\nMaple Leafs\nTOR\n13\n\n\n" + }, + { + "type": "Title", + "element_id": "98656277bdadc9ef7d1a9e1bc969579b", + "metadata": { + "data_source": { + "url": "gs://utic-test-ingest-fixtures/nested-2/stanley-cups.xlsx", + "version": "COul9MuE0/8CEAE=", + "record_locator": { + "protocol": "gs", + "remote_file_path": "utic-test-ingest-fixtures/nested-2/stanley-cups.xlsx" + }, + "date_created": "2023-06-20T23:48:24.973000+00:00", + "date_modified": "2023-06-20T23:48:24.973000+00:00" + }, + "filetype": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", + "languages": [ + "tur" + ], + "page_number": 2, + "page_name": "Stanley Cups Since 67", + "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR0
" }, - "text": "\n\n\nStanley Cups\nUnnamed: 1\nUnnamed: 2\n\n\n\n\nTeam\nLocation\nStanley Cups\n\n\nBlues\nSTL\n1\n\n\nFlyers\nPHI\n2\n\n\nMaple Leafs\nTOR\n13\n\n\n" + "text": "Stanley Cups Since 67" }, { "type": "Table", - "element_id": "0699dddf33814117e04654068f5182f6", + "element_id": "31421b5cd94fedb10dc82738503b4505", "metadata": { "data_source": { "url": "gs://utic-test-ingest-fixtures/nested-2/stanley-cups.xlsx", @@ -35,10 +86,13 @@ "date_modified": "2023-06-20T23:48:24.973000+00:00" }, "filetype": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", + "languages": [ + "tur" + ], "page_number": 2, "page_name": "Stanley Cups Since 67", - "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
Stanley Cups Since 67Unnamed: 1Unnamed: 2
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR0
" + "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
TeamLocationStanley Cups
BluesSTL1
FlyersPHI2
Maple LeafsTOR0
" }, - "text": "\n\n\nStanley Cups Since 67\nUnnamed: 1\nUnnamed: 2\n\n\n\n\nTeam\nLocation\nStanley Cups\n\n\nBlues\nSTL\n1\n\n\nFlyers\nPHI\n2\n\n\nMaple Leafs\nTOR\n0\n\n\n" + "text": "\n\n\nTeam\nLocation\nStanley Cups\n\n\nBlues\nSTL\n1\n\n\nFlyers\nPHI\n2\n\n\nMaple Leafs\nTOR\n0\n\n\n" } ] \ No newline at end of file diff --git a/test_unstructured_ingest/expected-structured-output/onedrive/utic-test-ingest-fixtures/tests-example.xls.json b/test_unstructured_ingest/expected-structured-output/onedrive/utic-test-ingest-fixtures/tests-example.xls.json index ee6e938c41..65d3b51c49 100644 --- a/test_unstructured_ingest/expected-structured-output/onedrive/utic-test-ingest-fixtures/tests-example.xls.json +++ b/test_unstructured_ingest/expected-structured-output/onedrive/utic-test-ingest-fixtures/tests-example.xls.json @@ -1,7 +1,7 @@ [ { "type": "Table", - "element_id": "0e2d044a26942328e2b8647574232e7f", + "element_id": "a5c9668a6055bca2865ea5e6d16ea1e0", "metadata": { "data_source": { "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", @@ -14,15 +14,64 @@ }, "filename": "tests-example.xls", "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], "page_number": 1, "page_name": "Example Test", - "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
MCWhat is 2+2?4correct3incorrectUnnamed: 6Unnamed: 7Unnamed: 8
MAWhat C datatypes are 8 bits? (assume i386)intfloatdoublechar
TFBagpipes are awesome.true
ESSHow have the original Henry Hornbostel buildings influenced campus architecture and design in the last 30 years?
ORDRank the following in their order of operation.ParenthesesExponentsDivisionAddition
FIBThe student activities fee is95dollars for students enrolled in19units or more,
MATMatch the lower-case greek letter with its capital form.λΛαγΓφΦ
" + "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
MCWhat is 2+2?4correct3incorrect
MAWhat C datatypes are 8 bits? (assume i386)intfloatdoublechar
TFBagpipes are awesome.true
ESSHow have the original Henry Hornbostel buildings influenced campus architecture and design in the last 30 years?
ORDRank the following in their order of operation.ParenthesesExponentsDivisionAddition
FIBThe student activities fee is95dollars for students enrolled in19units or more,
MATMatch the lower-case greek letter with its capital form.λΛαγΓφΦ
" }, - "text": "\n\n\nMC\nWhat is 2+2?\n4\ncorrect\n3\nincorrect\nUnnamed: 6\nUnnamed: 7\nUnnamed: 8\n\n\n\n\nMA\nWhat C datatypes are 8 bits? (assume i386)\nint\n\nfloat\n\ndouble\n\nchar\n\n\nTF\nBagpipes are awesome.\ntrue\n\n\n\n\n\n\n\n\nESS\nHow have the original Henry Hornbostel buildings influenced campus architecture and design in the last 30 years?\n\n\n\n\n\n\n\n\n\nORD\nRank the following in their order of operation.\nParentheses\nExponents\nDivision\nAddition\n\n\n\n\n\nFIB\nThe student activities fee is\n95\ndollars for students enrolled in\n19\nunits or more,\n\n\n\n\n\nMAT\nMatch the lower-case greek letter with its capital form.\nλ\nΛ\nα\nγ\nΓ\nφ\nΦ\n\n\n" + "text": "\n\n\nMC\nWhat is 2+2?\n4\ncorrect\n3\nincorrect\n\n\n\n\n\nMA\nWhat C datatypes are 8 bits? (assume i386)\nint\n\nfloat\n\ndouble\n\nchar\n\n\nTF\nBagpipes are awesome.\ntrue\n\n\n\n\n\n\n\n\nESS\nHow have the original Henry Hornbostel buildings influenced campus architecture and design in the last 30 years?\n\n\n\n\n\n\n\n\n\nORD\nRank the following in their order of operation.\nParentheses\nExponents\nDivision\nAddition\n\n\n\n\n\nFIB\nThe student activities fee is\n95\ndollars for students enrolled in\n19\nunits or more,\n\n\n\n\n\nMAT\nMatch the lower-case greek letter with its capital form.\nλ\nΛ\nα\nγ\nΓ\nφ\nΦ\n\n\n" + }, + { + "type": "Title", + "element_id": "1d34c23ff08573afa07b42842b41277a", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 2, + "page_name": "Format Abbr." + }, + "text": "http://www.cmu.edu/blackboard" + }, + { + "type": "Title", + "element_id": "05440c6ca94cb55f6d185d8bd92ce9d6", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 2, + "page_name": "Format Abbr." + }, + "text": "Question Format Abbreviations" }, { "type": "Table", - "element_id": "5c56dd4c5b649b873ebd848312e66753", + "element_id": "e39c724f1b09a4c3286b6368538e05fc", "metadata": { "data_source": { "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", @@ -35,15 +84,18 @@ }, "filename": "tests-example.xls", "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], "page_number": 2, "page_name": "Format Abbr.", - "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
Unnamed: 0Unnamed: 1
http://www.cmu.edu/blackboard
Question Format Abbreviations
AbbreviationQuestion Type
MCMultiple Choice
MAMultiple Answer
TFTrue/False
ESSEssay
ORDOrdering
MATMatching
FIBFill in the Blank
FILFile response
NUMNumeric Response
SRShort response
OPOpinion
FIB_PLUSMultiple Fill in the Blank
JUMBLED_SENTENCEJumbled Sentence
QUIZ_BOWLQuiz Bowl
" + "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
AbbreviationQuestion Type
MCMultiple Choice
MAMultiple Answer
TFTrue/False
ESSEssay
ORDOrdering
MATMatching
FIBFill in the Blank
FILFile response
NUMNumeric Response
SRShort response
OPOpinion
FIB_PLUSMultiple Fill in the Blank
JUMBLED_SENTENCEJumbled Sentence
QUIZ_BOWLQuiz Bowl
" }, - "text": "\n\n\nUnnamed: 0\nUnnamed: 1\n\n\n\n\n\n\n\n\n\n\n\n\nhttp://www.cmu.edu/blackboard\n\n\n\n\n\n\n\nQuestion Format Abbreviations\n\n\n\n\n\n\n\nAbbreviation\nQuestion Type\n\n\nMC\nMultiple Choice\n\n\nMA\nMultiple Answer\n\n\nTF\nTrue/False\n\n\nESS\nEssay\n\n\nORD\nOrdering\n\n\nMAT\nMatching\n\n\nFIB\nFill in the Blank\n\n\nFIL\nFile response\n\n\nNUM\nNumeric Response\n\n\nSR\nShort response\n\n\nOP\nOpinion\n\n\nFIB_PLUS\nMultiple Fill in the Blank\n\n\nJUMBLED_SENTENCE\nJumbled Sentence\n\n\nQUIZ_BOWL\nQuiz Bowl\n\n\n" + "text": "\n\n\nAbbreviation\nQuestion Type\n\n\nMC\nMultiple Choice\n\n\nMA\nMultiple Answer\n\n\nTF\nTrue/False\n\n\nESS\nEssay\n\n\nORD\nOrdering\n\n\nMAT\nMatching\n\n\nFIB\nFill in the Blank\n\n\nFIL\nFile response\n\n\nNUM\nNumeric Response\n\n\nSR\nShort response\n\n\nOP\nOpinion\n\n\nFIB_PLUS\nMultiple Fill in the Blank\n\n\nJUMBLED_SENTENCE\nJumbled Sentence\n\n\nQUIZ_BOWL\nQuiz Bowl\n\n\n" }, { - "type": "Table", - "element_id": "f48657c4eb70d98975e567248d0ef4bb", + "type": "Title", + "element_id": "1d34c23ff08573afa07b42842b41277a", "metadata": { "data_source": { "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", @@ -56,10 +108,291 @@ }, "filename": "tests-example.xls", "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], "page_number": 3, - "page_name": "Readme", - "text_as_html": "\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
Unnamed: 0
http://www.cmu.edu/blackboard
File Information
Source
http://www.cmu.edu/blackboard/files/evaluate/tests-example.xls
Version
1.0 (January 2012)
Contact
bb-help@andrew.cmu.edu
About
This is an example and template for preparing Blackboard tests offline. See the full directions at: http://www.cmu.edu/blackboard/evaluate#manage_tests/import_questions
" + "page_name": "Readme" + }, + "text": "http://www.cmu.edu/blackboard" + }, + { + "type": "Title", + "element_id": "85ada878f2345c23b8a74a931d2e20a4", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 3, + "page_name": "Readme" + }, + "text": "File Information" + }, + { + "type": "Title", + "element_id": "0e570ca6fabe24f94e52c1833f3ffd25", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 3, + "page_name": "Readme" + }, + "text": "Source" + }, + { + "type": "Title", + "element_id": "4cf4ff5597274d0c1ce8ae5a17ead4df", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 3, + "page_name": "Readme" + }, + "text": "http://www.cmu.edu/blackboard/files/evaluate/tests-example.xls" + }, + { + "type": "Title", + "element_id": "4cf4ff5597274d0c1ce8ae5a17ead4df", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel" + }, + "text": "http://www.cmu.edu/blackboard/files/evaluate/tests-example.xls" + }, + { + "type": "Title", + "element_id": "dd167905de0defcaf72de673ee44c074", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 3, + "page_name": "Readme" + }, + "text": "Version" + }, + { + "type": "UncategorizedText", + "element_id": "5f9d7b40d332fef76efdd0a97bcb8617", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 3, + "page_name": "Readme" + }, + "text": "1.0 (January 2012)" + }, + { + "type": "UncategorizedText", + "element_id": "5f9d7b40d332fef76efdd0a97bcb8617", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel" + }, + "text": "1.0 (January 2012)" + }, + { + "type": "Title", + "element_id": "2b5c3d26721ae9c350cf3009318b626f", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 3, + "page_name": "Readme" + }, + "text": "Contact" + }, + { + "type": "Title", + "element_id": "53d2273ac70fc31640cc45af840dbd42", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 3, + "page_name": "Readme" + }, + "text": "bb-help@andrew.cmu.edu" + }, + { + "type": "Title", + "element_id": "53d2273ac70fc31640cc45af840dbd42", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel" + }, + "text": "bb-help@andrew.cmu.edu" + }, + { + "type": "Title", + "element_id": "4efca0d10c5feb8e9b35eb1d994f2905", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 3, + "page_name": "Readme" + }, + "text": "About" + }, + { + "type": "NarrativeText", + "element_id": "4c9720f1540cc84d33e30e09aca8c077", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel", + "languages": [ + "eng" + ], + "page_number": 3, + "page_name": "Readme" + }, + "text": "This is an example and template for preparing Blackboard tests offline. See the full directions at: http://www.cmu.edu/blackboard/evaluate#manage_tests/import_questions" + }, + { + "type": "NarrativeText", + "element_id": "4c9720f1540cc84d33e30e09aca8c077", + "metadata": { + "data_source": { + "url": "/drives/b!3vfYDk3GHEaRbo1pkhLPIRXZrzTLHCtCm5WV6KY1m_0-lOjrjQaAS6X30Pv_E4VX/root:/utic-test-ingest-fixtures/tests-example.xls", + "record_locator": { + "user_pname": "devops@unstructuredio.onmicrosoft.com", + "server_relative_path": "utic-test-ingest-fixtures/tests-example.xls" + }, + "date_created": "2023-08-24T03:00:43", + "date_modified": "2023-08-24T03:00:43" + }, + "filename": "tests-example.xls", + "filetype": "application/vnd.ms-excel" }, - "text": "\n\n\nUnnamed: 0\n\n\n\n\n\n\n\n\n\n\nhttp://www.cmu.edu/blackboard\n\n\n\n\n\nFile Information\n\n\n\n\n\n\n\n\nSource\n\n\nhttp://www.cmu.edu/blackboard/files/evaluate/tests-example.xls\n\n\n\n\n\n\n\n\nVersion\n\n\n1.0 (January 2012)\n\n\n\n\n\n\n\n\nContact\n\n\nbb-help@andrew.cmu.edu\n\n\n\n\n\n\n\n\nAbout\n\n\nThis is an example and template for preparing Blackboard tests offline. See the full directions at: http://www.cmu.edu/blackboard/evaluate#manage_tests/import_questions\n\n\n" + "text": "This is an example and template for preparing Blackboard tests offline. See the full directions at: http://www.cmu.edu/blackboard/evaluate#manage_tests/import_questions" } ] \ No newline at end of file diff --git a/unstructured/partition/xlsx.py b/unstructured/partition/xlsx.py index ebffd6cdf9..871c002b3a 100644 --- a/unstructured/partition/xlsx.py +++ b/unstructured/partition/xlsx.py @@ -1,14 +1,20 @@ from tempfile import SpooledTemporaryFile -from typing import IO, BinaryIO, List, Optional, Union, cast +from typing import IO, Any, BinaryIO, Dict, List, Optional, Tuple, Union, cast +import numpy as np import pandas as pd from lxml.html.soupparser import fromstring as soupparser_fromstring from unstructured.chunking.title import add_chunking_strategy +from unstructured.cleaners.core import clean_bullets from unstructured.documents.elements import ( Element, ElementMetadata, + ListItem, + NarrativeText, Table, + Text, + Title, process_metadata, ) from unstructured.file_utils.filetype import FileType, add_metadata_with_filetype @@ -18,6 +24,13 @@ get_last_modified_date_from_file, spooled_to_bytes_io_if_needed, ) +from unstructured.partition.lang import detect_languages +from unstructured.partition.text_type import ( + is_bulleted_text, + is_possible_narrative_text, + is_possible_numbered_list, + is_possible_title, +) @process_metadata() @@ -28,8 +41,10 @@ def partition_xlsx( file: Optional[Union[IO[bytes], SpooledTemporaryFile]] = None, metadata_filename: Optional[str] = None, include_metadata: bool = True, + languages: List[str] = ["auto"], metadata_last_modified: Optional[str] = None, - include_header: bool = True, + include_header: bool = False, + find_subtable: bool = True, **kwargs, ) -> List[Element]: """Partitions Microsoft Excel Documents in .xlsx format into its document elements. @@ -42,43 +57,340 @@ def partition_xlsx( A file-like object using "rb" mode --> open(filename, "rb"). include_metadata Determines whether or not metadata is included in the output. + languages + The list of languages present in the document. metadata_last_modified The day of the last modification include_header Determines whether or not header info info is included in text and medatada.text_as_html """ exactly_one(filename=filename, file=file) + if not isinstance(languages, list): + raise TypeError( + 'The language parameter must be a list of language codes as strings, ex. ["eng"]', + ) last_modification_date = None + header = 0 if include_header else None + if filename: - sheets = pd.read_excel(filename, sheet_name=None) + sheets = pd.read_excel(filename, sheet_name=None, header=header) last_modification_date = get_last_modified_date(filename) elif file: f = spooled_to_bytes_io_if_needed( cast(Union[BinaryIO, SpooledTemporaryFile], file), ) - sheets = pd.read_excel(f, sheet_name=None) + sheets = pd.read_excel(f, sheet_name=None, header=header) last_modification_date = get_last_modified_date_from_file(file) elements: List[Element] = [] page_number = 0 - for sheet_name, table in sheets.items(): + for sheet_name, sheet in sheets.items(): page_number += 1 - html_text = table.to_html(index=False, header=include_header, na_rep="") - text = soupparser_fromstring(html_text).text_content() - - if include_metadata: - metadata = ElementMetadata( - text_as_html=html_text, - page_name=sheet_name, - page_number=page_number, - filename=metadata_filename or filename, - last_modified=metadata_last_modified or last_modification_date, - ) + if not find_subtable: + html_text = sheet.to_html(index=False, header=include_header, na_rep="") + text = soupparser_fromstring(html_text).text_content() + + if include_metadata: + metadata = ElementMetadata( + text_as_html=html_text, + page_name=sheet_name, + page_number=page_number, + filename=metadata_filename or filename, + last_modified=metadata_last_modified or last_modification_date, + ) + else: + metadata = ElementMetadata() + + table = Table(text=text, metadata=metadata) + elements.append(table) else: - metadata = ElementMetadata() + _connected_components = _get_connected_components(sheet) + for _connected_component, _min_max_coords in _connected_components: + min_x, min_y, max_x, max_y = _min_max_coords + + subtable = sheet.iloc[min_x : max_x + 1, min_y : max_y + 1] # noqa: E203 + single_non_empty_rows, single_non_empty_row_contents = _single_non_empty_rows( + subtable, + ) + ( + front_non_consecutive, + last_non_consecutive, + ) = _find_first_and_last_non_consecutive_row( + single_non_empty_rows, + subtable.shape, + ) + + metadata = _get_metadata( + include_metadata, + sheet_name, + page_number, + metadata_filename or filename, + metadata_last_modified or last_modification_date, + ) + + # NOTE(klaijan) - need to explicitly define the condition to avoid the case of 0 + if front_non_consecutive is not None and last_non_consecutive is not None: + first_row = int(front_non_consecutive - max_x) + last_row = int(max_x - last_non_consecutive) + subtable = _get_sub_subtable(subtable, (first_row, last_row)) - table = Table(text=text, metadata=metadata) - elements.append(table) + if front_non_consecutive is not None: + for content in single_non_empty_row_contents[: front_non_consecutive + 1]: + languages = detect_languages(str(content), languages) + element = _check_content_element_type(str(content)) + element.metadata = metadata + element.metadata.languages = languages + elements.append(element) + + if subtable is not None and len(subtable) == 1: + element = _check_content_element_type(str(subtable.iloc[0].values[0])) + elements.append(element) + + elif subtable is not None: + # parse subtables as html + html_text = subtable.to_html(index=False, header=include_header, na_rep="") + text = soupparser_fromstring(html_text).text_content() + languages = detect_languages(text, languages) + subtable = Table(text=text) + subtable.metadata = metadata + subtable.metadata.text_as_html = html_text + subtable.metadata.languages = languages + elements.append(subtable) + + if front_non_consecutive is not None and last_non_consecutive is not None: + for content in single_non_empty_row_contents[ + front_non_consecutive + 1 : # noqa: E203 + ]: + languages = detect_languages(str(content), languages) + element = _check_content_element_type(str(content)) + element.metadata = metadata + element.metadata.languages = languages + elements.append(element) return elements + + +def _get_connected_components( + sheet: pd.DataFrame, + filter: bool = True, +): + """ + Identify connected components of non-empty cells in an excel sheet. + + Args: + sheet: an excel sheet read in DataFrame. + filter (bool, optional): If True (default), filters out overlapping components + to return distinct components. + + Returns: + A list of tuples, each containing: + - A list of tuples representing the connected component's cell coordinates. + - A tuple with the min and max x and y coordinates bounding the connected component. + + Note: + This function performs a depth-first search (DFS) to identify connected components of + non-empty cells in the sheet. If 'filter' is set to True, it also filters out + overlapping components to return distinct components. + """ + max_row, max_col = sheet.shape + visited = set() + connected_components = [] + + def dfs(row, col, component): + if ( + row < 0 + or row >= sheet.shape[0] + or col < 0 + or col >= sheet.shape[1] + or (row, col) in visited + ): + return + visited.add((row, col)) + + if not pd.isna(sheet.iat[row, col]): + component.append((row, col)) + + # Explore neighboring cells + dfs(row - 1, col, component) # Above + dfs(row + 1, col, component) # Below + dfs(row, col - 1, component) # Left + dfs(row, col + 1, component) # Right + + for row in range(max_row): + for col in range(max_col): + if (row, col) not in visited and not pd.isna(sheet.iat[row, col]): + component: List[dict] = [] + dfs(row, col, component) + min_x, min_y, max_x, max_y = _find_min_max_coord(component) + connected_components.append( + { + "component": component, + "min_x": min_x, + "min_y": min_y, + "max_x": max_x, + "max_y": max_y, + }, + ) + if filter: + connected_components = _filter_overlapping_tables(connected_components) + return [ + ( + connected_component["component"], + ( + connected_component["min_x"], + connected_component["min_y"], + connected_component["max_x"], + connected_component["max_y"], + ), + ) + for connected_component in connected_components + ] + + +def _filter_overlapping_tables( + connected_components: List[Dict[Any, Any]], +) -> List[Dict[Any, Any]]: + """ + Filter out overlapping connected components to return distinct components. + """ + sorted_components = sorted(connected_components, key=lambda x: x["min_x"]) + merged_components: List[dict] = [] + current_component = None + for component in sorted_components: + if current_component is None: + current_component = component + else: + # Check if component overlaps with the current_component + if component["min_x"] <= current_component["max_x"]: + # Merge the components and update min_x, max_x + current_component["component"].extend(component["component"]) + current_component["min_x"] = min(current_component["min_x"], component["min_x"]) + current_component["max_x"] = max(current_component["max_x"], component["max_x"]) + current_component["min_y"] = min(current_component["min_y"], component["min_y"]) + current_component["max_y"] = max(current_component["max_y"], component["max_y"]) + else: + # No overlap, add the current_component to the merged list + merged_components.append(current_component) + # Update the current_component + current_component = component + # Append the last current_component to the merged list + if current_component is not None: + merged_components.append(current_component) + return merged_components + + +def _find_min_max_coord( + connected_component: List[Dict[Any, Any]], +) -> Tuple[Union[int, float], Union[int, float], Union[int, float], Union[int, float]]: + """ + Find the minimum and maximum coordinates (bounding box) of a connected component. + """ + min_x, min_y, max_x, max_y = float("inf"), float("inf"), float("-inf"), float("-inf") + for _x, _y in connected_component: + if _x < min_x: + min_x = _x + if _y < min_y: + min_y = _y + if _x > max_x: + max_x = _x + if _y > max_y: + max_y = _y + return min_x, min_y, max_x, max_y + + +def _get_sub_subtable(subtable: pd.DataFrame, first_and_last_row: Tuple[int, int]) -> pd.DataFrame: + """ + Extract a sub-subtable from a given subtable based on the first and last row range. + """ + # TODO(klaijan) - to further check for sub subtable, we could check whether + # two consecutive rows contains full row of cells. + # if yes, it might not be a header. We should check the length. + first_row, last_row = first_and_last_row + if last_row == first_row: + return None + return subtable.iloc[first_row : last_row + 1] # noqa: E203 + + +def _find_first_and_last_non_consecutive_row( + row_indices: List[int], + table_shape: Tuple[int, int], +) -> Tuple[Optional[int], Optional[int]]: + """ + Find the indices of the first and last non-consecutive rows in a list of row indices. + """ + # If the table is a single column with one or more rows + table_rows, table_cols = table_shape + if len(row_indices) == 1 or (len(row_indices) == table_rows and table_cols == 1): + return row_indices[0], row_indices[0] + + arr = np.array(row_indices) + front_non_consecutive = next( + (i for i, (x, y) in enumerate(zip(arr, arr[1:])) if x + 1 != y), + None, + ) + reversed_arr = arr[::-1] # Reverse the array + last_non_consecutive = next( + (i for i, (x, y) in enumerate(zip(reversed_arr, reversed_arr[1:])) if x - 1 != y), + None, + ) + return front_non_consecutive, last_non_consecutive + + +def _single_non_empty_rows(subtable) -> Tuple[List[int], List[str]]: + """ + Identify single non-empty rows in a subtable and extract their row indices and contents. + """ + single_non_empty_rows = [] + single_non_empty_row_contents = [] + for index, row in subtable.iterrows(): + if row.count() == 1: + single_non_empty_rows.append(index) + single_non_empty_row_contents.append(row.dropna().iloc[0]) + return single_non_empty_rows, single_non_empty_row_contents + + +def _check_content_element_type(text: str) -> Element: + """ + Classify the type of content element based on its text. + """ + if is_bulleted_text(text): + return ListItem( + text=clean_bullets(text), + ) + elif is_possible_numbered_list(text): + return ListItem( + text=text, + ) + elif is_possible_narrative_text(text): + return NarrativeText( + text=text, + ) + elif is_possible_title(text): + return Title( + text=text, + ) + else: + return Text( + text=text, + ) + + +def _get_metadata( + include_metadata: bool = True, + sheet_name: Optional[str] = None, + page_number: Optional[int] = -1, + filename: Optional[str] = None, + last_modification_date: Union[str, None] = None, +) -> ElementMetadata: + """Returns metadata depending on `include_metadata` flag""" + if include_metadata: + metadata = ElementMetadata( + page_name=sheet_name, + page_number=page_number, + filename=filename, + last_modified=last_modification_date, + ) + else: + metadata = ElementMetadata() + return metadata