From bb0f46ea9105a1ea8fc6cc1306d8812a3e319b09 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Thu, 19 Sep 2024 11:05:40 +0700 Subject: [PATCH 01/18] feat: update stale action --- .github/workflows/stale.yaml | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) create mode 100644 .github/workflows/stale.yaml diff --git a/.github/workflows/stale.yaml b/.github/workflows/stale.yaml new file mode 100644 index 000000000..fdf0105fa --- /dev/null +++ b/.github/workflows/stale.yaml @@ -0,0 +1,26 @@ +name: Close stale issues +on: + schedule: + - cron: "0 0 * * *" # Runs at 00:00 UTC every day + +jobs: + stale: + runs-on: ubuntu-latest + steps: + - uses: actions/stale@v9 + with: + repo-token: ${{ secrets.GITHUB_TOKEN }} + + stale-issue-message: | + 'This issue is stale because it has not received any activities recently.' + stale-pr-message: | + 'This issue is stale because it has not received any activities recently.' + close-issue-message: | + 'This issue was closed because it has been stalled with no activity.' + + days-before-issue-stale: 1 + days-before-issue-close: 0 + days-before-pr-stale: 90 + days-before-pr-close: -1 + exempt-issue-labels: "documentation,tutorial,TODO" + operations-per-run: 300 # The maximum number of operations per run, used to control rate limiting. From 93ec97268db70ea82e0ebab0fdfe16ff9547c087 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Thu, 26 Sep 2024 15:57:25 +0700 Subject: [PATCH 02/18] feat: comfort gocr2 format --- libs/kotaemon/kotaemon/loaders/ocr_loader.py | 24 ++++++++++++++------ 1 file changed, 17 insertions(+), 7 deletions(-) diff --git a/libs/kotaemon/kotaemon/loaders/ocr_loader.py b/libs/kotaemon/kotaemon/loaders/ocr_loader.py index 4e009deef..71d5ce3e9 100644 --- a/libs/kotaemon/kotaemon/loaders/ocr_loader.py +++ b/libs/kotaemon/kotaemon/loaders/ocr_loader.py @@ -10,12 +10,12 @@ from kotaemon.base import Document -from .utils.pdf_ocr import parse_ocr_output, read_pdf_unstructured -from .utils.table import strip_special_chars_markdown +# from .utils.pdf_ocr import parse_ocr_output, read_pdf_unstructured +# from .utils.table import strip_special_chars_markdown logger = logging.getLogger(__name__) -DEFAULT_OCR_ENDPOINT = "http://127.0.0.1:8000/v2/ai/infer/" +DEFAULT_OCR_ENDPOINT = "http://localhost:8881/ai/infer/" @retry( @@ -25,8 +25,8 @@ ) def tenacious_api_post(url, file_path, table_only, **kwargs): with file_path.open("rb") as content: - files = {"input": content} - data = {"job_id": uuid4(), "table_only": table_only} + files = {"file": content} + data = {"ocr_type": "ocr"} resp = requests.post(url=url, files=files, data=data, **kwargs) resp.raise_for_status() return resp @@ -178,16 +178,26 @@ def load_data( resp = tenacious_api_post( url=self.ocr_endpoint, file_path=file_path, table_only=False ) - ocr_results = resp.json()["result"] + ocr_results = [resp.json()["result"]] extra_info = extra_info or {} result = [] for ocr_result in ocr_results: result.append( Document( - content=ocr_result["csv_string"], + content=ocr_result, metadata=extra_info, ) ) return result + +if __name__ == '__main__': + file_path = "/home/kan/projects/nlp/kotaemon/00167_Mizuho.png" + + with open(file_path, "rb") as content: + files = {"file": content} + data = {"ocr_type": "ocr"} + resp = requests.post(url="http://127.0.0.1:8881/ai/infer/", files=files, data=data,) + resp.raise_for_status() + From 4b7e9ca96648138f6b0335ac1e8f2a41a5840ef4 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Sun, 15 Dec 2024 14:04:01 +0700 Subject: [PATCH 03/18] fix: resolve conflicts --- .../indices/ingests/extension_manager.py | 150 ++++++++++++++++++ .../kotaemon/indices/ingests/files.py | 8 +- libs/kotaemon/kotaemon/loaders/__init__.py | 3 +- libs/kotaemon/kotaemon/loaders/ocr_loader.py | 96 +++++++++-- libs/ktem/ktem/app.py | 4 + libs/ktem/ktem/index/file/pipelines.py | 8 +- libs/ktem/ktem/pages/chat/__init__.py | 8 +- libs/ktem/ktem/pages/settings.py | 49 ++++++ libs/ktem/ktem/settings.py | 6 + 9 files changed, 310 insertions(+), 22 deletions(-) create mode 100644 libs/kotaemon/kotaemon/indices/ingests/extension_manager.py diff --git a/libs/kotaemon/kotaemon/indices/ingests/extension_manager.py b/libs/kotaemon/kotaemon/indices/ingests/extension_manager.py new file mode 100644 index 000000000..69a923187 --- /dev/null +++ b/libs/kotaemon/kotaemon/indices/ingests/extension_manager.py @@ -0,0 +1,150 @@ +from copy import deepcopy + +from decouple import config +from llama_index.core.readers.base import BaseReader +from theflow.settings import settings as flowsettings + +from kotaemon.loaders import ( + AdobeReader, + AzureAIDocumentIntelligenceLoader, + DirectoryReader, + HtmlReader, + MathpixPDFReader, + MhtmlReader, + OCRReader, + PandasExcelReader, + PDFThumbnailReader, + TxtReader, + UnstructuredReader, + ImageReader, + GOCR2ImageReader +) + + +unstructured = UnstructuredReader() +adobe_reader = AdobeReader() +azure_reader = AzureAIDocumentIntelligenceLoader( + endpoint=str(config("AZURE_DI_ENDPOINT", default="")), + credential=str(config("AZURE_DI_CREDENTIAL", default="")), + cache_dir=getattr(flowsettings, "KH_MARKDOWN_OUTPUT_DIR", None), +) +adobe_reader.vlm_endpoint = azure_reader.vlm_endpoint = getattr( + flowsettings, "KH_VLM_ENDPOINT", "" +) + +KH_DEFAULT_FILE_EXTRACTORS: dict[str, BaseReader] = { + ".xlsx": PandasExcelReader(), + ".docx": unstructured, + ".pptx": unstructured, + ".xls": unstructured, + ".doc": unstructured, + ".html": HtmlReader(), + ".mhtml": MhtmlReader(), + ".png": ImageReader(), + ".jpeg": ImageReader(), + ".jpg": ImageReader(), + ".tiff": unstructured, + ".tif": unstructured, + ".pdf": PDFThumbnailReader(), + ".txt": TxtReader(), + ".md": TxtReader(), +} + + +class ExtensionManager: + """Pool of loaders for extensions""" + def __init__(self): + self._supported, self._default_index = self._init_supported() + + def get_current_loader(self) -> dict[str, BaseReader]: + return deepcopy({k: self.get_selected_loader_by_extension(k)[0] for k, _ in self._supported.items()}) + + @staticmethod + def _init_supported() -> tuple[dict[str, list[BaseReader]], dict[str, str]]: + supported: dict[str, list[BaseReader]] = { + ".xlsx": [PandasExcelReader()], + ".docx": [unstructured], + ".pptx": [unstructured], + ".xls": [unstructured], + ".doc": [unstructured], + ".html": [HtmlReader()], + ".mhtml": [MhtmlReader()], + ".png": [GOCR2ImageReader(), unstructured], + ".jpeg": [GOCR2ImageReader(), unstructured], + ".jpg": [GOCR2ImageReader(), unstructured], + ".tiff": [unstructured], + ".tif": [unstructured], + ".pdf": [PDFThumbnailReader()], + ".txt": [TxtReader()], + ".md": [TxtReader()], + } + + default_index = { + k: ExtensionManager.get_loader_name(vs[0]) + for k, vs + in supported.items() + } + + return supported, default_index + + def load(self, settings: dict, prefix="extension"): + for key, value in settings.items(): + if not key.startswith(prefix): + continue + extension = key.replace("extension.", "") + if extension in self._supported: + # Update the default index + # Only if it's in supported list + supported_loader_names = self.get_loaders_by_extension(extension)[1] + if value in supported_loader_names: + self._default_index[extension] = value + else: + print(f"[{extension}]Can not find loader: {value} from list of " + f"supported extensions: {supported_loader_names}") + + @staticmethod + def get_loader_name(loader: BaseReader) -> str: + return loader.__class__.__name__ + + def get_supported_extensions(self): + return list(self._supported.keys()) + + def get_loaders_by_extension(self, extension: str) -> tuple[list[BaseReader], list[str]]: + loaders = self._supported[extension] + loaders_name = [self.get_loader_name(loader) for loader in loaders] + return loaders, loaders_name + + def get_selected_loader_by_extension(self, extension: str) -> tuple[BaseReader, str]: + supported_loaders: list[BaseReader] = self._supported[extension] + + for loader in supported_loaders: + loader_name = self.get_loader_name(loader) + + if loader_name == self._default_index[extension]: + return loader, loader_name + + raise Exception(f"can not find the selected loader for extension: {extension}") + + def generate_gradio_settings(self) -> dict[str, dict]: + """Generates the settings dictionary for use in Gradio.""" + settings = {} + + for extension, loaders in self._supported.items(): + current_loader: str = self._default_index[extension] + loaders_choices: list[str] = [self.get_loader_name(loader) for loader in loaders] + + settings[extension] = { + "name": f"Loader {extension}", + "value": current_loader, + "choices": loaders_choices, + "component": "dropdown", # You can customize this to "radio" if needed + } + + return settings + + +extension_manager = ExtensionManager() + + +if __name__ == "__main__": + print(extension_manager.get_loaders_by_extension(".xlsx")) diff --git a/libs/kotaemon/kotaemon/indices/ingests/files.py b/libs/kotaemon/kotaemon/indices/ingests/files.py index 18db7ca86..e6254dd65 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/files.py +++ b/libs/kotaemon/kotaemon/indices/ingests/files.py @@ -23,6 +23,8 @@ TxtReader, UnstructuredReader, WebReader, + UnstructuredReader, + ImageReader, ) web_reader = WebReader() @@ -47,9 +49,9 @@ ".doc": unstructured, ".html": HtmlReader(), ".mhtml": MhtmlReader(), - ".png": unstructured, - ".jpeg": unstructured, - ".jpg": unstructured, + ".png": ImageReader(), + ".jpeg": ImageReader(), + ".jpg": ImageReader(), ".tiff": unstructured, ".tif": unstructured, ".pdf": PDFThumbnailReader(), diff --git a/libs/kotaemon/kotaemon/loaders/__init__.py b/libs/kotaemon/kotaemon/loaders/__init__.py index f498da806..18bf997d7 100644 --- a/libs/kotaemon/kotaemon/loaders/__init__.py +++ b/libs/kotaemon/kotaemon/loaders/__init__.py @@ -7,7 +7,7 @@ from .excel_loader import ExcelReader, PandasExcelReader from .html_loader import HtmlReader, MhtmlReader from .mathpix_loader import MathpixPDFReader -from .ocr_loader import ImageReader, OCRReader +from .ocr_loader import ImageReader, OCRReader, GOCR2ImageReader from .pdf_loader import PDFThumbnailReader from .txt_loader import TxtReader from .unstructured_loader import UnstructuredReader @@ -32,4 +32,5 @@ "PDFThumbnailReader", "WebReader", "DoclingReader", + "GOCR2ImageReader" ] diff --git a/libs/kotaemon/kotaemon/loaders/ocr_loader.py b/libs/kotaemon/kotaemon/loaders/ocr_loader.py index 71d5ce3e9..416e57370 100644 --- a/libs/kotaemon/kotaemon/loaders/ocr_loader.py +++ b/libs/kotaemon/kotaemon/loaders/ocr_loader.py @@ -10,12 +10,12 @@ from kotaemon.base import Document -# from .utils.pdf_ocr import parse_ocr_output, read_pdf_unstructured -# from .utils.table import strip_special_chars_markdown +from .utils.pdf_ocr import parse_ocr_output, read_pdf_unstructured +from .utils.table import strip_special_chars_markdown logger = logging.getLogger(__name__) -DEFAULT_OCR_ENDPOINT = "http://localhost:8881/ai/infer/" +DEFAULT_OCR_ENDPOINT = "http://127.0.0.1:8000/v2/ai/infer/" @retry( @@ -25,8 +25,8 @@ ) def tenacious_api_post(url, file_path, table_only, **kwargs): with file_path.open("rb") as content: - files = {"file": content} - data = {"ocr_type": "ocr"} + files = {"input": content} + data = {"job_id": uuid4(), "table_only": table_only} resp = requests.post(url=url, files=files, data=data, **kwargs) resp.raise_for_status() return resp @@ -178,26 +178,94 @@ def load_data( resp = tenacious_api_post( url=self.ocr_endpoint, file_path=file_path, table_only=False ) - ocr_results = [resp.json()["result"]] + ocr_results = resp.json()["result"] extra_info = extra_info or {} result = [] for ocr_result in ocr_results: result.append( Document( - content=ocr_result, + content=ocr_result["csv_string"], metadata=extra_info, ) ) return result -if __name__ == '__main__': - file_path = "/home/kan/projects/nlp/kotaemon/00167_Mizuho.png" - with open(file_path, "rb") as content: - files = {"file": content} - data = {"ocr_type": "ocr"} - resp = requests.post(url="http://127.0.0.1:8881/ai/infer/", files=files, data=data,) - resp.raise_for_status() +class GOCR2ImageReader(BaseReader): + default_endpoint = "http://localhost:8881/ai/infer/" + """Read Image using GOCR-2.0 + + Args: + endpoint: URL to GOCR endpoint. If not provided, will look for + environment variable `GOCR2_ENDPOINT` or use the default + (http://localhost:8881/ai/infer/) + """ + + def __init__(self, endpoint: Optional[str] = None): + """Init the OCR reader with OCR endpoint (FullOCR pipeline)""" + super().__init__() + self.endpoint = endpoint or os.getenv( + "GOCR2_ENDPOINT", self.default_endpoint + ) + + def load_data( + self, + file_path: Path, + extra_info: dict | None = None, + **kwargs + ) -> List[Document]: + """Load data using OCR reader + + Args: + file_path (Path): Path to PDF file + extra_info (Path): Extra information while inference + + Returns: + List[Document]: list of documents extracted from the PDF file + """ + @retry( + stop=stop_after_attempt(6), + wait=wait_exponential(multiplier=20, exp_base=2, min=1, max=1000), + after=after_log(logger, logging.WARNING), + ) + def _tenacious_api_post( + url: str, + file_path: str, + ocr_type: str = "ocr", + **kwargs + ): + with file_path.open("rb") as content: + files = {"file": content} + data = {"ocr_type": ocr_type} + resp = requests.post(url=url, files=files, data=data, **kwargs) + resp.raise_for_status() + return resp + + file_path = Path(file_path).resolve() + + # call the API from GOCR endpoint + if "response_content" in kwargs: + # overriding response content if specified + ocr_results = kwargs["response_content"] + else: + # call original API + resp = _tenacious_api_post( + url=self.endpoint, + file_path=file_path + ) + ocr_results = [resp.json()["result"]] + + extra_info = extra_info or {} + result = [] + for ocr_result in ocr_results: + result.append( + Document( + content=ocr_result, + metadata=extra_info, + ) + ) + + return result diff --git a/libs/ktem/ktem/app.py b/libs/ktem/ktem/app.py index 7142377e1..236777940 100644 --- a/libs/ktem/ktem/app.py +++ b/libs/ktem/ktem/app.py @@ -3,6 +3,9 @@ import gradio as gr import pluggy +from aiohttp.web_fileresponse import extension + +from kotaemon.indices.ingests.extension_manager import extension_manager from ktem import extension_protocol from ktem.assets import PDFJS_PREBUILT_DIR, KotaemonTheme from ktem.components import reasonings @@ -63,6 +66,7 @@ def __init__(self): self.default_settings = SettingGroup( application=BaseSettingGroup(settings=settings.SETTINGS_APP), reasoning=SettingReasoningGroup(settings=settings.SETTINGS_REASONING), + extension=BaseSettingGroup(settings=extension_manager.generate_gradio_settings()) ) self._callbacks: dict[str, list] = {} diff --git a/libs/ktem/ktem/index/file/pipelines.py b/libs/ktem/ktem/index/file/pipelines.py index 4d53e6538..6198c9416 100644 --- a/libs/ktem/ktem/index/file/pipelines.py +++ b/libs/ktem/ktem/index/file/pipelines.py @@ -14,6 +14,9 @@ from typing import Generator, Optional, Sequence import tiktoken + +from kotaemon.indices.ingests.extension_manager import extension_manager +from kotaemon.loaders import OCRReader from ktem.db.models import engine from ktem.embeddings.manager import embedding_models_manager from ktem.llms.manager import llms @@ -36,7 +39,7 @@ from kotaemon.embeddings import BaseEmbeddings from kotaemon.indices import VectorIndexing, VectorRetrieval from kotaemon.indices.ingests.files import ( - KH_DEFAULT_FILE_EXTRACTORS, + # KH_DEFAULT_FILE_EXTRACTORS, adobe_reader, azure_reader, docling_reader, @@ -671,7 +674,8 @@ class IndexDocumentPipeline(BaseFileIndexIndexing): @Param.auto(depends_on="reader_mode") def readers(self): - readers = deepcopy(KH_DEFAULT_FILE_EXTRACTORS) + # readers = deepcopy(KH_DEFAULT_FILE_EXTRACTORS) + readers: dict[str, BaseReader] = extension_manager.get_current_loader() print("reader_mode", self.reader_mode) if self.reader_mode == "adobe": readers[".pdf"] = adobe_reader diff --git a/libs/ktem/ktem/pages/chat/__init__.py b/libs/ktem/ktem/pages/chat/__init__.py index 86ed46fa3..279747bc9 100644 --- a/libs/ktem/ktem/pages/chat/__init__.py +++ b/libs/ktem/ktem/pages/chat/__init__.py @@ -5,6 +5,9 @@ from typing import Optional import gradio as gr +from filelock import FileLock + +from kotaemon.indices.ingests.extension_manager import extension_manager from ktem.app import BasePage from ktem.components import reasonings from ktem.db.models import Conversation, engine @@ -20,7 +23,7 @@ from theflow.settings import settings as flowsettings from kotaemon.base import Document -from kotaemon.indices.ingests.files import KH_DEFAULT_FILE_EXTRACTORS +# from kotaemon.indices.ingests.files import KH_DEFAULT_FILE_EXTRACTORS from ...utils import SUPPORTED_LANGUAGE_MAP, get_file_names_regex from .chat_panel import ChatPanel @@ -160,7 +163,8 @@ def on_building_ui(self): if len(self._app.index_manager.indices) > 0: with gr.Accordion(label="Quick Upload") as _: self.quick_file_upload = File( - file_types=list(KH_DEFAULT_FILE_EXTRACTORS.keys()), + # file_types=list(KH_DEFAULT_FILE_EXTRACTORS.keys()), + file_types=extension_manager.get_supported_extensions(), file_count="multiple", container=True, show_label=False, diff --git a/libs/ktem/ktem/pages/settings.py b/libs/ktem/ktem/pages/settings.py index b74d641f0..75442045a 100644 --- a/libs/ktem/ktem/pages/settings.py +++ b/libs/ktem/ktem/pages/settings.py @@ -1,6 +1,8 @@ import hashlib import gradio as gr + +from kotaemon.indices.ingests.extension_manager import extension_manager from ktem.app import BasePage from ktem.components import reasonings from ktem.db.models import Settings, User, engine @@ -113,6 +115,7 @@ def on_building_ui(self): self.app_tab() self.index_tab() self.reasoning_tab() + self.extension_tab() self.setting_save_btn = gr.Button( "Save changes", variant="primary", scale=1, elem_classes=["right-button"] @@ -177,7 +180,12 @@ def on_register_events(self): self.save_setting, inputs=[self._user_id] + self.components(), outputs=self._settings_state, + ).then( + fn=lambda state: extension_manager.load(state), + inputs=[self._settings_state], + outputs=None ) + self._components["reasoning.use"].change( self.change_reasoning_mode, inputs=[self._components["reasoning.use"]], @@ -282,6 +290,42 @@ def index_tab(self): if si.special_type == "embedding": self._embeddings.append(obj) + def extension_tab(self): + extensions: list[str] = list(self._default_settings.extension.settings.keys()) + + lefts = extensions[::2] + rights = extensions[1::2] + + if len(lefts) > len(rights): + rights += [""] + + assert len(lefts) == len(rights) + + with gr.Tab("Extension settings"): + for left, right in zip(lefts, rights): + left_setting = self._default_settings.extension.settings.get(left, None) + right_setting = self._default_settings.extension.settings.get(right, None) + + with gr.Row(): + with gr.Column(1): + if left_setting: + left_gradio_obj = render_setting_item( + left_setting, + left_setting.value + ) + self._components[f"extension.{left}"] = left_gradio_obj + + with gr.Column(1): + if right_setting: + right_gradio_obj = render_setting_item( + right_setting, + right_setting.value + ) + self._components[f"extension.{right}"] = right_gradio_obj + else: + gr.TextArea(value="", visible=False) + + def reasoning_tab(self): with gr.Tab("Reasoning settings", visible=self._render_reasoning_tab): with gr.Group(): @@ -387,6 +431,10 @@ def _on_app_created(self): inputs=self._user_id, outputs=[self._settings_state] + self.components(), show_progress="hidden", + ).then( + fn=lambda state: extension_manager.load(state), + inputs=[self._settings_state], + outputs=None ) def update_llms(): @@ -416,6 +464,7 @@ def update_embeddings(): outputs=[llm], show_progress="hidden", ) + for emb in self._embeddings: self._app.app.load( update_embeddings, diff --git a/libs/ktem/ktem/settings.py b/libs/ktem/ktem/settings.py index 89f5c6518..ac9d897db 100644 --- a/libs/ktem/ktem/settings.py +++ b/libs/ktem/ktem/settings.py @@ -2,6 +2,8 @@ from pydantic import BaseModel, Field +from kotaemon.indices.ingests.extension_manager import extension_manager + class SettingItem(BaseModel): """Represent a setting item @@ -124,6 +126,7 @@ class SettingGroup(BaseModel): application: BaseSettingGroup = Field(default_factory=BaseSettingGroup) index: SettingIndexGroup = Field(default_factory=SettingIndexGroup) reasoning: SettingReasoningGroup = Field(default_factory=SettingReasoningGroup) + extension: BaseSettingGroup = Field(default_factory=BaseSettingGroup) def flatten(self) -> dict: """Render the setting group into value""" @@ -137,6 +140,9 @@ def flatten(self) -> dict: for key, value in self.reasoning.flatten().items(): output[f"reasoning.{key}"] = value + for key, value in self.extension.flatten().items(): + output[f"extension.{key}"] = value + return output def get_setting_item(self, path: str) -> SettingItem: From 84e5683de28e3c3f4cb8df9e8b182f5ac4460ee8 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Fri, 25 Oct 2024 07:53:56 +0000 Subject: [PATCH 04/18] feat: resolve conflicts --- libs/kotaemon/kotaemon/loaders/ocr_loader.py | 1 + 1 file changed, 1 insertion(+) diff --git a/libs/kotaemon/kotaemon/loaders/ocr_loader.py b/libs/kotaemon/kotaemon/loaders/ocr_loader.py index 416e57370..90c67732b 100644 --- a/libs/kotaemon/kotaemon/loaders/ocr_loader.py +++ b/libs/kotaemon/kotaemon/loaders/ocr_loader.py @@ -269,3 +269,4 @@ def _tenacious_api_post( ) return result + \ No newline at end of file From 81577336bb648f7ff76785de743b41f61ca38927 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Wed, 2 Oct 2024 14:08:13 +0000 Subject: [PATCH 05/18] feat: update state for extension manager while load setting --- .../kotaemon/indices/ingests/files.py | 37 +++++++++--------- libs/kotaemon/kotaemon/loaders/test.jpg | Bin 0 -> 91668 bytes libs/ktem/ktem/pages/settings.py | 6 +-- 3 files changed, 21 insertions(+), 22 deletions(-) create mode 100644 libs/kotaemon/kotaemon/loaders/test.jpg diff --git a/libs/kotaemon/kotaemon/indices/ingests/files.py b/libs/kotaemon/kotaemon/indices/ingests/files.py index e6254dd65..3cbdc8c7a 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/files.py +++ b/libs/kotaemon/kotaemon/indices/ingests/files.py @@ -26,6 +26,7 @@ UnstructuredReader, ImageReader, ) +from libs.kotaemon.kotaemon.indices.ingests.extension_manager import extension_manager web_reader = WebReader() unstructured = UnstructuredReader() @@ -41,23 +42,23 @@ ) = docling_reader.vlm_endpoint = getattr(flowsettings, "KH_VLM_ENDPOINT", "") -KH_DEFAULT_FILE_EXTRACTORS: dict[str, BaseReader] = { - ".xlsx": PandasExcelReader(), - ".docx": unstructured, - ".pptx": unstructured, - ".xls": unstructured, - ".doc": unstructured, - ".html": HtmlReader(), - ".mhtml": MhtmlReader(), - ".png": ImageReader(), - ".jpeg": ImageReader(), - ".jpg": ImageReader(), - ".tiff": unstructured, - ".tif": unstructured, - ".pdf": PDFThumbnailReader(), - ".txt": TxtReader(), - ".md": TxtReader(), -} +# KH_DEFAULT_FILE_EXTRACTORS: dict[str, BaseReader] = { +# ".xlsx": PandasExcelReader(), +# ".docx": unstructured, +# ".pptx": unstructured, +# ".xls": unstructured, +# ".doc": unstructured, +# ".html": HtmlReader(), +# ".mhtml": MhtmlReader(), +# ".png": ImageReader(), +# ".jpeg": ImageReader(), +# ".jpg": ImageReader(), +# ".tiff": unstructured, +# ".tif": unstructured, +# ".pdf": PDFThumbnailReader(), +# ".txt": TxtReader(), +# ".md": TxtReader(), +# } class DocumentIngestor(BaseComponent): @@ -92,7 +93,7 @@ class DocumentIngestor(BaseComponent): def _get_reader(self, input_files: list[str | Path]): """Get appropriate readers for the input files based on file extension""" file_extractors: dict[str, BaseReader] = { - ext: reader for ext, reader in KH_DEFAULT_FILE_EXTRACTORS.items() + ext: reader for ext, reader in extension_manager.get_current_loader().items() } for ext, cls in self.override_file_extractors.items(): file_extractors[ext] = cls() diff --git a/libs/kotaemon/kotaemon/loaders/test.jpg b/libs/kotaemon/kotaemon/loaders/test.jpg new file mode 100644 index 0000000000000000000000000000000000000000..5d933abcdaa65b7e07e726d5576bf7c0435cd081 GIT binary patch literal 91668 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

XYkXEM=?wvGoo9_=RAjPoL(K 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%0D|(%hM3%f@%qzvH;?i!0uJ$3k9xU zrpkEuvS-d5w5ut^*8EP-I*+0dCtRcpOqZ!wCbn(q>O&@~h5AyC5C^ygX;KnmSke#4 z4Ra^29BWyL{u6CBHc}w0#{*laNzYHd_SUq1s2+JcanT}dRI_z^{S55WkxzksgydAR zHK7e54?aMu>q{XbyZZrTnV*ltu*?PA51xA%WX08ib0A`7gV?!yH<|nDo&57a%UxA3 zH0$U9p+6yOw3ckrvzbSawo&p^Pc3`K{{rYb4Z5f#aVq}FjqlosGex7`*@*`CqmDHG z)I2(u+2`hjWVjPdbn!6_l=~#7Qr;-;@#qarQu*2%Hp3UZ3b*t^$ExJN0^Z#ev@o~}nQ){14-hFgk%HZ%lJv4N7;Jj(Oqo60I+fNqr4wmdu;w_8& zZV?k)8V0hWMk*^n=gug%n+(Oyp>I-=!sOnT5!A2V zEsMKkJ2?hgMgi~`3DL~laUlM5;PEroYVC+7WeXMhIlAE+BZDRz>j!@kp^oFwyNCR_ zY6L?{Dh4vE0JE?S3byja``#AAg+5rzaCWwM-S{GsV_ycaK`UYT2FmD$3%h<-O;ake z9a^H^0>02uUrmg<;Hi`%3YFv<@6fR44;jrFdd^r9@q-k&}Sq%4j$ z!yYDfo#;n$ocNQaJ~qf21M8ZpO4L<)Cqht(5`^h;C+n(>dXEPE7VqR43L`a@&NNGN~{4=pJi+1?xUQ7&i57c3#i)R|!I;xn(Qy4fqag?wJu7Fj#pONUwg&$^!zGoSmUVS%%NSQ&a_Db5a<5F< zk#ukM>!v^@%4(!@flsb$LC(SdT+*3m)Lfh8iH#~1$`bd0a_~+;N~u+G&{nHRpEw7u z6}eW_Qupyr9dOAj38e}ZI)B;TJg+|w%3_F`k(j*S-njB+H`4ZwHF?KDV7nzjkDKxq(r2mM zuQvQvh%!+}`f;WcA3|ZcV)`!pNE~5g;Sm;Dix1%KNauu`$TRP zMeXrIpAXwGa^P|J$|@MR4&t5m)-mQ0U#7|ujEG9%j}DP2v2_bNBV{^%)AhX9ImEBP zZU+p^n1`ciZ*yq6)5Xgp<_28ytM#R`mpp^0YcY_>F^_wY?_$LqFVF^0S=x8^Yx`4M z8({Zjoky>H3M4b3^(_`XeH$R8%ri!zV*5MZsAh8Sai^Jv>Tu$Q@~hBRiU=-_Nzv=YO$`q?Oy`BZ5uw; zqbTNKG0Fj)iinOa-;)5jtefmkN@!o8QF4y`Mu$k~7b*+VWZd!aFBwyaBW_4s+$aQ5!e6o|r zq?!_^x5jdLbIX0>l|xfqxIq>EN19PxRrJim2ra_E?C;9I(& z(x7uiYTUmE+VG&lT5>cD}(m#09g2@9L2`usIu2GHeG^|0{Zpp zMnkw1vv6Cs{qo~S7*E#56u|jbn+y=PTOJdmZ*xbz?2x&>9u+v7eQ9dt`r-20y$;lz z1yLZfw<*@fn$;eWC>!}KDS|lk?ev*b*{=W`Fjn#UXwCglz@JdN;uWMuM!EG_k`1v;0SIY2&8+2oK5?HHeYs!VWtZ(=ng=a2YGMVUXNutZi8F5|O%A&u^n(na z0LWLs*gp1ffVX@+i}9^0kbDhDsM8n8f^DVm+{yc^e;$$X&38SKkErMy%f<~hUp57Cla{fuxOX+_Qu$9 zs((dcVV-(nF}f`8&JRCbzIhGDcM^`gUwAG|kb0;^se8c{jO{ruCf(S6rhT)^Ae-;l z&O6JA3wmMSead&(!0_!A6dzjIt@4wReidI`Mq@9G_lm?N{&X2meb_DOK@k6^iH1Kt zQm+k2byTk|Eb@=eo!i4D0$^k5mCZX>hFsRCfzt$ zLF23bP-aQn0tZE*qiwqM%o}5eH-I18 z{KGfCEamate>p}CaHjR#r>>4ivpd>JZ-qh2*C*BD$689M#F(Mr@hvEQpAd^sLzHQ1 z=?6PHwr+&aPpea%^O1pd(INEOrh@8s{oP+*;7^&|hi+9>!J{4}Zx7?Noa1G#c@_R( zz{EAW#M1#dp>@b6wG9jER;99= z+yWxRqA8Z)n9CjHo?1s0rr_T&?X#e;CgK*gV#iOKE}F~fKMb=TU7%jDA^Nh1E8F}^*P%(ml}8Ld9cCsHAn^H}TIj_dQtHn1^zu%pFymJ_Q1?l#mA=sB z_bg%yR^t~dmAkLi>rk*LxYO5hsn(I-!$n|tQC+;e>bE;e0!(x@$~;WJU%tFtoqZX! zY(FM~1+J2Y1QQwm!OJdnd**@Nwfip6u-M^6nJ9GCcm`uJS*~k7$`8LevkiIGVxTm%qAeNwQAL( zvT>g?)g4z>40Gk%RFVrUFU814wtRsm&wv<_-lsYvd!I$GG$=462~)uFt_P;qikCoL z{0_7{xi|1u8r Date: Sun, 15 Dec 2024 14:04:37 +0700 Subject: [PATCH 06/18] fix: resolve conflicts --- .../{extension_manager.py => extensions.py} | 38 +++++++++++-------- .../kotaemon/indices/ingests/files.py | 6 +-- libs/kotaemon/kotaemon/loaders/__init__.py | 2 +- libs/kotaemon/kotaemon/loaders/ocr_loader.py | 19 ++-------- libs/ktem/ktem/app.py | 9 +++-- libs/ktem/ktem/index/file/pipelines.py | 8 +--- libs/ktem/ktem/pages/chat/__init__.py | 7 ++-- libs/ktem/ktem/pages/settings.py | 17 ++++----- libs/ktem/ktem/settings.py | 2 - 9 files changed, 50 insertions(+), 58 deletions(-) rename libs/kotaemon/kotaemon/indices/ingests/{extension_manager.py => extensions.py} (81%) diff --git a/libs/kotaemon/kotaemon/indices/ingests/extension_manager.py b/libs/kotaemon/kotaemon/indices/ingests/extensions.py similarity index 81% rename from libs/kotaemon/kotaemon/indices/ingests/extension_manager.py rename to libs/kotaemon/kotaemon/indices/ingests/extensions.py index 69a923187..cff13819b 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/extension_manager.py +++ b/libs/kotaemon/kotaemon/indices/ingests/extensions.py @@ -7,20 +7,16 @@ from kotaemon.loaders import ( AdobeReader, AzureAIDocumentIntelligenceLoader, - DirectoryReader, + GOCR2ImageReader, HtmlReader, - MathpixPDFReader, + ImageReader, MhtmlReader, - OCRReader, PandasExcelReader, PDFThumbnailReader, TxtReader, UnstructuredReader, - ImageReader, - GOCR2ImageReader ) - unstructured = UnstructuredReader() adobe_reader = AdobeReader() azure_reader = AzureAIDocumentIntelligenceLoader( @@ -53,11 +49,17 @@ class ExtensionManager: """Pool of loaders for extensions""" + def __init__(self): self._supported, self._default_index = self._init_supported() def get_current_loader(self) -> dict[str, BaseReader]: - return deepcopy({k: self.get_selected_loader_by_extension(k)[0] for k, _ in self._supported.items()}) + return deepcopy( + { + k: self.get_selected_loader_by_extension(k)[0] + for k, _ in self._supported.items() + } + ) @staticmethod def _init_supported() -> tuple[dict[str, list[BaseReader]], dict[str, str]]: @@ -80,9 +82,7 @@ def _init_supported() -> tuple[dict[str, list[BaseReader]], dict[str, str]]: } default_index = { - k: ExtensionManager.get_loader_name(vs[0]) - for k, vs - in supported.items() + k: ExtensionManager.get_loader_name(vs[0]) for k, vs in supported.items() } return supported, default_index @@ -99,8 +99,10 @@ def load(self, settings: dict, prefix="extension"): if value in supported_loader_names: self._default_index[extension] = value else: - print(f"[{extension}]Can not find loader: {value} from list of " - f"supported extensions: {supported_loader_names}") + print( + f"[{extension}]Can not find loader: {value} from list of " + f"supported extensions: {supported_loader_names}" + ) @staticmethod def get_loader_name(loader: BaseReader) -> str: @@ -109,12 +111,16 @@ def get_loader_name(loader: BaseReader) -> str: def get_supported_extensions(self): return list(self._supported.keys()) - def get_loaders_by_extension(self, extension: str) -> tuple[list[BaseReader], list[str]]: + def get_loaders_by_extension( + self, extension: str + ) -> tuple[list[BaseReader], list[str]]: loaders = self._supported[extension] loaders_name = [self.get_loader_name(loader) for loader in loaders] return loaders, loaders_name - def get_selected_loader_by_extension(self, extension: str) -> tuple[BaseReader, str]: + def get_selected_loader_by_extension( + self, extension: str + ) -> tuple[BaseReader, str]: supported_loaders: list[BaseReader] = self._supported[extension] for loader in supported_loaders: @@ -131,7 +137,9 @@ def generate_gradio_settings(self) -> dict[str, dict]: for extension, loaders in self._supported.items(): current_loader: str = self._default_index[extension] - loaders_choices: list[str] = [self.get_loader_name(loader) for loader in loaders] + loaders_choices: list[str] = [ + self.get_loader_name(loader) for loader in loaders + ] settings[extension] = { "name": f"Loader {extension}", diff --git a/libs/kotaemon/kotaemon/indices/ingests/files.py b/libs/kotaemon/kotaemon/indices/ingests/files.py index 3cbdc8c7a..23d1c3a4c 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/files.py +++ b/libs/kotaemon/kotaemon/indices/ingests/files.py @@ -16,7 +16,6 @@ DoclingReader, HtmlReader, MathpixPDFReader, - MhtmlReader, OCRReader, PandasExcelReader, PDFThumbnailReader, @@ -26,7 +25,7 @@ UnstructuredReader, ImageReader, ) -from libs.kotaemon.kotaemon.indices.ingests.extension_manager import extension_manager +from libs.kotaemon.kotaemon.indices.ingests.extensions import extension_manager web_reader = WebReader() unstructured = UnstructuredReader() @@ -93,7 +92,8 @@ class DocumentIngestor(BaseComponent): def _get_reader(self, input_files: list[str | Path]): """Get appropriate readers for the input files based on file extension""" file_extractors: dict[str, BaseReader] = { - ext: reader for ext, reader in extension_manager.get_current_loader().items() + ext: reader + for ext, reader in extension_manager.get_current_loader().items() } for ext, cls in self.override_file_extractors.items(): file_extractors[ext] = cls() diff --git a/libs/kotaemon/kotaemon/loaders/__init__.py b/libs/kotaemon/kotaemon/loaders/__init__.py index 18bf997d7..b6c8dc198 100644 --- a/libs/kotaemon/kotaemon/loaders/__init__.py +++ b/libs/kotaemon/kotaemon/loaders/__init__.py @@ -7,7 +7,7 @@ from .excel_loader import ExcelReader, PandasExcelReader from .html_loader import HtmlReader, MhtmlReader from .mathpix_loader import MathpixPDFReader -from .ocr_loader import ImageReader, OCRReader, GOCR2ImageReader +from .ocr_loader import GOCR2ImageReader, ImageReader, OCRReader from .pdf_loader import PDFThumbnailReader from .txt_loader import TxtReader from .unstructured_loader import UnstructuredReader diff --git a/libs/kotaemon/kotaemon/loaders/ocr_loader.py b/libs/kotaemon/kotaemon/loaders/ocr_loader.py index 90c67732b..f996ca275 100644 --- a/libs/kotaemon/kotaemon/loaders/ocr_loader.py +++ b/libs/kotaemon/kotaemon/loaders/ocr_loader.py @@ -206,15 +206,10 @@ class GOCR2ImageReader(BaseReader): def __init__(self, endpoint: Optional[str] = None): """Init the OCR reader with OCR endpoint (FullOCR pipeline)""" super().__init__() - self.endpoint = endpoint or os.getenv( - "GOCR2_ENDPOINT", self.default_endpoint - ) + self.endpoint = endpoint or os.getenv("GOCR2_ENDPOINT", self.default_endpoint) def load_data( - self, - file_path: Path, - extra_info: dict | None = None, - **kwargs + self, file_path: Path, extra_info: dict | None = None, **kwargs ) -> List[Document]: """Load data using OCR reader @@ -232,10 +227,7 @@ def load_data( after=after_log(logger, logging.WARNING), ) def _tenacious_api_post( - url: str, - file_path: str, - ocr_type: str = "ocr", - **kwargs + url: str, file_path: str, ocr_type: str = "ocr", **kwargs ): with file_path.open("rb") as content: files = {"file": content} @@ -252,10 +244,7 @@ def _tenacious_api_post( ocr_results = kwargs["response_content"] else: # call original API - resp = _tenacious_api_post( - url=self.endpoint, - file_path=file_path - ) + resp = _tenacious_api_post(url=self.endpoint, file_path=file_path) ocr_results = [resp.json()["result"]] extra_info = extra_info or {} diff --git a/libs/ktem/ktem/app.py b/libs/ktem/ktem/app.py index 236777940..53ec58d80 100644 --- a/libs/ktem/ktem/app.py +++ b/libs/ktem/ktem/app.py @@ -3,9 +3,6 @@ import gradio as gr import pluggy -from aiohttp.web_fileresponse import extension - -from kotaemon.indices.ingests.extension_manager import extension_manager from ktem import extension_protocol from ktem.assets import PDFJS_PREBUILT_DIR, KotaemonTheme from ktem.components import reasonings @@ -15,6 +12,8 @@ from theflow.settings import settings from theflow.utils.modules import import_dotted_string +from kotaemon.indices.ingests.extensions import extension_manager + class BaseApp: """The main app of Kotaemon @@ -66,7 +65,9 @@ def __init__(self): self.default_settings = SettingGroup( application=BaseSettingGroup(settings=settings.SETTINGS_APP), reasoning=SettingReasoningGroup(settings=settings.SETTINGS_REASONING), - extension=BaseSettingGroup(settings=extension_manager.generate_gradio_settings()) + extension=BaseSettingGroup( + settings=extension_manager.generate_gradio_settings() + ), ) self._callbacks: dict[str, list] = {} diff --git a/libs/ktem/ktem/index/file/pipelines.py b/libs/ktem/ktem/index/file/pipelines.py index 6198c9416..e74f1af0e 100644 --- a/libs/ktem/ktem/index/file/pipelines.py +++ b/libs/ktem/ktem/index/file/pipelines.py @@ -7,16 +7,12 @@ import time import warnings from collections import defaultdict -from copy import deepcopy from functools import lru_cache from hashlib import sha256 from pathlib import Path from typing import Generator, Optional, Sequence import tiktoken - -from kotaemon.indices.ingests.extension_manager import extension_manager -from kotaemon.loaders import OCRReader from ktem.db.models import engine from ktem.embeddings.manager import embedding_models_manager from ktem.llms.manager import llms @@ -38,8 +34,8 @@ from kotaemon.base import BaseComponent, Document, Node, Param, RetrievedDocument from kotaemon.embeddings import BaseEmbeddings from kotaemon.indices import VectorIndexing, VectorRetrieval -from kotaemon.indices.ingests.files import ( - # KH_DEFAULT_FILE_EXTRACTORS, +from kotaemon.indices.ingests.extensions import extension_manager +from kotaemon.indices.ingests.files import ( # KH_DEFAULT_FILE_EXTRACTORS, adobe_reader, azure_reader, docling_reader, diff --git a/libs/ktem/ktem/pages/chat/__init__.py b/libs/ktem/ktem/pages/chat/__init__.py index 279747bc9..c9f11bc47 100644 --- a/libs/ktem/ktem/pages/chat/__init__.py +++ b/libs/ktem/ktem/pages/chat/__init__.py @@ -6,8 +6,6 @@ import gradio as gr from filelock import FileLock - -from kotaemon.indices.ingests.extension_manager import extension_manager from ktem.app import BasePage from ktem.components import reasonings from ktem.db.models import Conversation, engine @@ -23,7 +21,7 @@ from theflow.settings import settings as flowsettings from kotaemon.base import Document -# from kotaemon.indices.ingests.files import KH_DEFAULT_FILE_EXTRACTORS +from kotaemon.indices.ingests.extensions import extension_manager from ...utils import SUPPORTED_LANGUAGE_MAP, get_file_names_regex from .chat_panel import ChatPanel @@ -31,6 +29,9 @@ from .control import ConversationControl from .report import ReportIssue +# from kotaemon.indices.ingests.files import KH_DEFAULT_FILE_EXTRACTORS + + DEFAULT_SETTING = "(default)" INFO_PANEL_SCALES = {True: 8, False: 4} diff --git a/libs/ktem/ktem/pages/settings.py b/libs/ktem/ktem/pages/settings.py index 9ada30556..001dcaef7 100644 --- a/libs/ktem/ktem/pages/settings.py +++ b/libs/ktem/ktem/pages/settings.py @@ -1,13 +1,13 @@ import hashlib import gradio as gr - -from kotaemon.indices.ingests.extension_manager import extension_manager from ktem.app import BasePage from ktem.components import reasonings from ktem.db.models import Settings, User, engine from sqlmodel import Session, select +from kotaemon.indices.ingests.extensions import extension_manager + signout_js = """ function(u, c, pw, pwc) { removeFromStorage('username'); @@ -183,7 +183,7 @@ def on_register_events(self): ).then( fn=lambda state: extension_manager.load(state), inputs=[self._settings_state], - outputs=None + outputs=None, ) self._components["reasoning.use"].change( @@ -304,28 +304,27 @@ def extension_tab(self): with gr.Tab("Extension settings"): for left, right in zip(lefts, rights): left_setting = self._default_settings.extension.settings.get(left, None) - right_setting = self._default_settings.extension.settings.get(right, None) + right_setting = self._default_settings.extension.settings.get( + right, None + ) with gr.Row(): with gr.Column(1): if left_setting: left_gradio_obj = render_setting_item( - left_setting, - left_setting.value + left_setting, left_setting.value ) self._components[f"extension.{left}"] = left_gradio_obj with gr.Column(1): if right_setting: right_gradio_obj = render_setting_item( - right_setting, - right_setting.value + right_setting, right_setting.value ) self._components[f"extension.{right}"] = right_gradio_obj else: gr.TextArea(value="", visible=False) - def reasoning_tab(self): with gr.Tab("Reasoning settings", visible=self._render_reasoning_tab): with gr.Group(): diff --git a/libs/ktem/ktem/settings.py b/libs/ktem/ktem/settings.py index ac9d897db..b9dea27d5 100644 --- a/libs/ktem/ktem/settings.py +++ b/libs/ktem/ktem/settings.py @@ -2,8 +2,6 @@ from pydantic import BaseModel, Field -from kotaemon.indices.ingests.extension_manager import extension_manager - class SettingItem(BaseModel): """Represent a setting item From e6b44396050b1d6c4f898dd58d96360c2d7eda45 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Wed, 2 Oct 2024 21:31:03 +0700 Subject: [PATCH 07/18] feat: comfort pre-commit --- .../kotaemon/indices/ingests/extensions.py | 22 ++----------------- .../kotaemon/indices/ingests/files.py | 21 +----------------- libs/kotaemon/kotaemon/loaders/ocr_loader.py | 2 +- 3 files changed, 4 insertions(+), 41 deletions(-) diff --git a/libs/kotaemon/kotaemon/indices/ingests/extensions.py b/libs/kotaemon/kotaemon/indices/ingests/extensions.py index cff13819b..f2d4ae34a 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/extensions.py +++ b/libs/kotaemon/kotaemon/indices/ingests/extensions.py @@ -1,4 +1,5 @@ from copy import deepcopy +from typing import Any from decouple import config from llama_index.core.readers.base import BaseReader @@ -9,7 +10,6 @@ AzureAIDocumentIntelligenceLoader, GOCR2ImageReader, HtmlReader, - ImageReader, MhtmlReader, PandasExcelReader, PDFThumbnailReader, @@ -28,24 +28,6 @@ flowsettings, "KH_VLM_ENDPOINT", "" ) -KH_DEFAULT_FILE_EXTRACTORS: dict[str, BaseReader] = { - ".xlsx": PandasExcelReader(), - ".docx": unstructured, - ".pptx": unstructured, - ".xls": unstructured, - ".doc": unstructured, - ".html": HtmlReader(), - ".mhtml": MhtmlReader(), - ".png": ImageReader(), - ".jpeg": ImageReader(), - ".jpg": ImageReader(), - ".tiff": unstructured, - ".tif": unstructured, - ".pdf": PDFThumbnailReader(), - ".txt": TxtReader(), - ".md": TxtReader(), -} - class ExtensionManager: """Pool of loaders for extensions""" @@ -131,7 +113,7 @@ def get_selected_loader_by_extension( raise Exception(f"can not find the selected loader for extension: {extension}") - def generate_gradio_settings(self) -> dict[str, dict]: + def generate_gradio_settings(self) -> dict[str, Any]: """Generates the settings dictionary for use in Gradio.""" settings = {} diff --git a/libs/kotaemon/kotaemon/indices/ingests/files.py b/libs/kotaemon/kotaemon/indices/ingests/files.py index 23d1c3a4c..1b93cf673 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/files.py +++ b/libs/kotaemon/kotaemon/indices/ingests/files.py @@ -8,6 +8,7 @@ from kotaemon.base import BaseComponent, Document, Param from kotaemon.indices.extractors import BaseDocParser +from kotaemon.indices.ingests.extensions import extension_manager from kotaemon.indices.splitters import BaseSplitter, TokenSplitter from kotaemon.loaders import ( AdobeReader, @@ -25,7 +26,6 @@ UnstructuredReader, ImageReader, ) -from libs.kotaemon.kotaemon.indices.ingests.extensions import extension_manager web_reader = WebReader() unstructured = UnstructuredReader() @@ -41,25 +41,6 @@ ) = docling_reader.vlm_endpoint = getattr(flowsettings, "KH_VLM_ENDPOINT", "") -# KH_DEFAULT_FILE_EXTRACTORS: dict[str, BaseReader] = { -# ".xlsx": PandasExcelReader(), -# ".docx": unstructured, -# ".pptx": unstructured, -# ".xls": unstructured, -# ".doc": unstructured, -# ".html": HtmlReader(), -# ".mhtml": MhtmlReader(), -# ".png": ImageReader(), -# ".jpeg": ImageReader(), -# ".jpg": ImageReader(), -# ".tiff": unstructured, -# ".tif": unstructured, -# ".pdf": PDFThumbnailReader(), -# ".txt": TxtReader(), -# ".md": TxtReader(), -# } - - class DocumentIngestor(BaseComponent): """Ingest common office document types into Document for indexing diff --git a/libs/kotaemon/kotaemon/loaders/ocr_loader.py b/libs/kotaemon/kotaemon/loaders/ocr_loader.py index f996ca275..51ef1d58a 100644 --- a/libs/kotaemon/kotaemon/loaders/ocr_loader.py +++ b/libs/kotaemon/kotaemon/loaders/ocr_loader.py @@ -227,7 +227,7 @@ def load_data( after=after_log(logger, logging.WARNING), ) def _tenacious_api_post( - url: str, file_path: str, ocr_type: str = "ocr", **kwargs + url: str, file_path: Path, ocr_type: str = "ocr", **kwargs ): with file_path.open("rb") as content: files = {"file": content} From 02f8a8198c9a9ba21340097d0b10baed8da7c8c0 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Wed, 2 Oct 2024 14:58:21 +0000 Subject: [PATCH 08/18] feat: update ocr loader --- libs/kotaemon/kotaemon/loaders/ocr_loader.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/libs/kotaemon/kotaemon/loaders/ocr_loader.py b/libs/kotaemon/kotaemon/loaders/ocr_loader.py index 51ef1d58a..6d9a35d29 100644 --- a/libs/kotaemon/kotaemon/loaders/ocr_loader.py +++ b/libs/kotaemon/kotaemon/loaders/ocr_loader.py @@ -250,10 +250,14 @@ def _tenacious_api_post( extra_info = extra_info or {} result = [] for ocr_result in ocr_results: + metadata = {"file_name": file_path.name, "page_label": 1} + if extra_info is not None: + metadata.update(extra_info) + result.append( Document( content=ocr_result, - metadata=extra_info, + metadata=metadata, ) ) From fca8a10101ce1cbf7559c69cc6a7ccd08e951d89 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Wed, 2 Oct 2024 15:05:57 +0000 Subject: [PATCH 09/18] feat: change default loader for image --- libs/kotaemon/kotaemon/indices/ingests/extensions.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/libs/kotaemon/kotaemon/indices/ingests/extensions.py b/libs/kotaemon/kotaemon/indices/ingests/extensions.py index f2d4ae34a..d24bd074a 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/extensions.py +++ b/libs/kotaemon/kotaemon/indices/ingests/extensions.py @@ -53,9 +53,9 @@ def _init_supported() -> tuple[dict[str, list[BaseReader]], dict[str, str]]: ".doc": [unstructured], ".html": [HtmlReader()], ".mhtml": [MhtmlReader()], - ".png": [GOCR2ImageReader(), unstructured], - ".jpeg": [GOCR2ImageReader(), unstructured], - ".jpg": [GOCR2ImageReader(), unstructured], + ".png": [unstructured, GOCR2ImageReader()], + ".jpeg": [unstructured, GOCR2ImageReader()], + ".jpg": [unstructured, GOCR2ImageReader()], ".tiff": [unstructured], ".tif": [unstructured], ".pdf": [PDFThumbnailReader()], From d9b9bc13fb948393930fad3610a4fffb0b9c8457 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Wed, 2 Oct 2024 22:30:59 +0700 Subject: [PATCH 10/18] feat: update guideline --- integration/got-ocr2.md | 30 ++++++++++++++++++++++++++++++ 1 file changed, 30 insertions(+) create mode 100644 integration/got-ocr2.md diff --git a/integration/got-ocr2.md b/integration/got-ocr2.md new file mode 100644 index 000000000..f569ddb4b --- /dev/null +++ b/integration/got-ocr2.md @@ -0,0 +1,30 @@ +## Extension Manager and GOT-OCR2.0 Loader + +## Key Features + +### 1. **GOCR2 as Image Reader** + +- **GOCR2ImageReader** is a new class designed to read images using the [**GOCR-2.0** OCR engine](https://github.com/Ucas-HaoranWei/GOT-OCR2.0). +- This reader is initialized with an endpoint that defaults to `http://localhost:8881/ai/infer/` for the OCR service, but can be configured through an environment variable `GOCR2_ENDPOINT` or passed explicitly. +- It uses exponential backoff retry mechanisms to ensure robustness during API calls. +- Supports loading image files and extracting their text content, returning structured document data. + +#### Setup + +- We provide the docker image, with fastapi for serving the GOT-OCR2.0. Pull the image from: + +```bash +docker run -d --gpus all -p 8881:8881 ghcr.io/phv2312/got-ocr2.0:main +``` + +- Detail implementation is placed at [ocr_loader.py](/libs/kotaemon/kotaemon/loaders/ocr_loader.py) + +### 2. **Extension Manager** + +- ExtensionManager allows users to dynamically manage multiple loaders for different file types. + +- Users can switch between multiple loaders for the same file extension, such as using the GOCR2ImageReader or a + different unstructured data parser for .png files. This provides the flexibility to choose the best-suited loader for the task at hand. + +- To change the default loader, go to **Settings**, then **Extension settings**. It displays a grid of extensions and + its supported loaders. Any modification will be saved to DB as other settings do. From dfc416e6f9f106c1166bac9c098a3c1a205f7ee0 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Wed, 2 Oct 2024 22:37:45 +0700 Subject: [PATCH 11/18] feat: update github stales & remove unncessary files --- .github/workflows/stale.yaml | 13 ++++++------- libs/kotaemon/kotaemon/loaders/test.jpg | Bin 91668 -> 0 bytes 2 files changed, 6 insertions(+), 7 deletions(-) delete mode 100644 libs/kotaemon/kotaemon/loaders/test.jpg diff --git a/.github/workflows/stale.yaml b/.github/workflows/stale.yaml index fdf0105fa..3ad174752 100644 --- a/.github/workflows/stale.yaml +++ b/.github/workflows/stale.yaml @@ -12,14 +12,13 @@ jobs: repo-token: ${{ secrets.GITHUB_TOKEN }} stale-issue-message: | - 'This issue is stale because it has not received any activities recently.' - stale-pr-message: | - 'This issue is stale because it has not received any activities recently.' - close-issue-message: | - 'This issue was closed because it has been stalled with no activity.' + Hi folk 😊, it looks like this issue has been inactive for a while. Is there any update or further action + needed? If not, we might consider closing it soon to keep our board clean and focused. - days-before-issue-stale: 1 - days-before-issue-close: 0 + But don't worry, you can reopen it anytime when needed.Thank you for your contributions to Kotaemon🪴 + + days-before-issue-stale: 30 + days-before-issue-close: 3 days-before-pr-stale: 90 days-before-pr-close: -1 exempt-issue-labels: "documentation,tutorial,TODO" diff --git a/libs/kotaemon/kotaemon/loaders/test.jpg b/libs/kotaemon/kotaemon/loaders/test.jpg deleted file mode 100644 index 5d933abcdaa65b7e07e726d5576bf7c0435cd081..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 91668 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

XYkXEM=?wvGoo9_=RAjPoL(K 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%0D|(%hM3%f@%qzvH;?i!0uJ$3k9xU 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 Date: Sat, 26 Oct 2024 04:11:15 +0000 Subject: [PATCH 12/18] feat: update extentions for pdf --- libs/kotaemon/kotaemon/indices/ingests/extensions.py | 10 ++++++---- libs/kotaemon/kotaemon/indices/ingests/files.py | 5 ----- libs/kotaemon/kotaemon/loaders/ocr_loader.py | 1 - libs/ktem/ktem/pages/chat/__init__.py | 1 - 4 files changed, 6 insertions(+), 11 deletions(-) diff --git a/libs/kotaemon/kotaemon/indices/ingests/extensions.py b/libs/kotaemon/kotaemon/indices/ingests/extensions.py index d24bd074a..0d956b96b 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/extensions.py +++ b/libs/kotaemon/kotaemon/indices/ingests/extensions.py @@ -45,6 +45,8 @@ def get_current_loader(self) -> dict[str, BaseReader]: @staticmethod def _init_supported() -> tuple[dict[str, list[BaseReader]], dict[str, str]]: + gocr = GOCR2ImageReader() + supported: dict[str, list[BaseReader]] = { ".xlsx": [PandasExcelReader()], ".docx": [unstructured], @@ -53,12 +55,12 @@ def _init_supported() -> tuple[dict[str, list[BaseReader]], dict[str, str]]: ".doc": [unstructured], ".html": [HtmlReader()], ".mhtml": [MhtmlReader()], - ".png": [unstructured, GOCR2ImageReader()], - ".jpeg": [unstructured, GOCR2ImageReader()], - ".jpg": [unstructured, GOCR2ImageReader()], + ".png": [unstructured, gocr], + ".jpeg": [unstructured, gocr], + ".jpg": [unstructured, gocr], ".tiff": [unstructured], ".tif": [unstructured], - ".pdf": [PDFThumbnailReader()], + ".pdf": [PDFThumbnailReader(), adobe_reader, azure_reader], ".txt": [TxtReader()], ".md": [TxtReader()], } diff --git a/libs/kotaemon/kotaemon/indices/ingests/files.py b/libs/kotaemon/kotaemon/indices/ingests/files.py index 1b93cf673..a2695598b 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/files.py +++ b/libs/kotaemon/kotaemon/indices/ingests/files.py @@ -18,13 +18,8 @@ HtmlReader, MathpixPDFReader, OCRReader, - PandasExcelReader, - PDFThumbnailReader, - TxtReader, UnstructuredReader, WebReader, - UnstructuredReader, - ImageReader, ) web_reader = WebReader() diff --git a/libs/kotaemon/kotaemon/loaders/ocr_loader.py b/libs/kotaemon/kotaemon/loaders/ocr_loader.py index 6d9a35d29..da10b8500 100644 --- a/libs/kotaemon/kotaemon/loaders/ocr_loader.py +++ b/libs/kotaemon/kotaemon/loaders/ocr_loader.py @@ -262,4 +262,3 @@ def _tenacious_api_post( ) return result - \ No newline at end of file diff --git a/libs/ktem/ktem/pages/chat/__init__.py b/libs/ktem/ktem/pages/chat/__init__.py index c9f11bc47..970706591 100644 --- a/libs/ktem/ktem/pages/chat/__init__.py +++ b/libs/ktem/ktem/pages/chat/__init__.py @@ -5,7 +5,6 @@ from typing import Optional import gradio as gr -from filelock import FileLock from ktem.app import BasePage from ktem.components import reasonings from ktem.db.models import Conversation, engine From a358fd25217ce86481b65eb274cd228254cddcf8 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Sat, 26 Oct 2024 04:14:30 +0000 Subject: [PATCH 13/18] feat: introduce docker-compose --- docker-compose.dev.yml | 9 +++++++++ docker-compose.yml | 30 ++++++++++++++++++++++++++++++ 2 files changed, 39 insertions(+) create mode 100644 docker-compose.dev.yml create mode 100644 docker-compose.yml diff --git a/docker-compose.dev.yml b/docker-compose.dev.yml new file mode 100644 index 000000000..d23cd4071 --- /dev/null +++ b/docker-compose.dev.yml @@ -0,0 +1,9 @@ +services: + kotaemon: + volumes: + - "./ktem_app_data:/app/ktem_app_data" + - "./libs/kotaemon:/app/kotaemon" + - "./libs/ktem:/app/ktem" + - "./flowsettings.py:/app/flowsettings.py" + ports: + - "7860:7860" diff --git a/docker-compose.yml b/docker-compose.yml new file mode 100644 index 000000000..ca873dc22 --- /dev/null +++ b/docker-compose.yml @@ -0,0 +1,30 @@ +services: + kotaemon: + build: + context: . + target: lite + dockerfile: Dockerfile + env_file: .env + environment: + - GRADIO_SERVER_NAME=0.0.0.0 + - GRADIO_SERVER_PORT=7860 + ports: + - "7860:7860" + networks: + - backend + # gocr: + # image: ghcr.io/phv2312/got-ocr2.0:main + # ports: + # - "8881:8881" + # deploy: + # resources: + # reservations: + # devices: + # - driver: nvidia + # count: 1 + # capabilities: [gpu] + # networks: + # - backend +networks: + backend: + driver: bridge From 609a1f0a2b744b44e7595358d9fb78627fcec0e3 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Sun, 15 Dec 2024 14:09:24 +0700 Subject: [PATCH 14/18] chore: refactor by pre-commit --- libs/kotaemon/kotaemon/indices/ingests/files.py | 1 - libs/kotaemon/kotaemon/loaders/__init__.py | 2 +- 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/libs/kotaemon/kotaemon/indices/ingests/files.py b/libs/kotaemon/kotaemon/indices/ingests/files.py index a2695598b..f289567b7 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/files.py +++ b/libs/kotaemon/kotaemon/indices/ingests/files.py @@ -15,7 +15,6 @@ AzureAIDocumentIntelligenceLoader, DirectoryReader, DoclingReader, - HtmlReader, MathpixPDFReader, OCRReader, UnstructuredReader, diff --git a/libs/kotaemon/kotaemon/loaders/__init__.py b/libs/kotaemon/kotaemon/loaders/__init__.py index b6c8dc198..05edbf663 100644 --- a/libs/kotaemon/kotaemon/loaders/__init__.py +++ b/libs/kotaemon/kotaemon/loaders/__init__.py @@ -32,5 +32,5 @@ "PDFThumbnailReader", "WebReader", "DoclingReader", - "GOCR2ImageReader" + "GOCR2ImageReader", ] From ae56308d5a896b87de8c2995d7ec3014140994af Mon Sep 17 00:00:00 2001 From: phv2312 Date: Sun, 15 Dec 2024 15:35:39 +0700 Subject: [PATCH 15/18] feat: update docling reader into extension manager --- .../kotaemon/indices/ingests/files.py | 2 +- libs/ktem/ktem/app.py | 3 +- libs/ktem/ktem/extensions/__init__.py | 0 .../ktem/extensions}/extensions.py | 125 +++++++++++++----- libs/ktem/ktem/index/file/pipelines.py | 23 ++-- libs/ktem/ktem/pages/chat/__init__.py | 2 +- libs/ktem/ktem/pages/settings.py | 3 +- 7 files changed, 105 insertions(+), 53 deletions(-) create mode 100644 libs/ktem/ktem/extensions/__init__.py rename libs/{kotaemon/kotaemon/indices/ingests => ktem/ktem/extensions}/extensions.py (51%) diff --git a/libs/kotaemon/kotaemon/indices/ingests/files.py b/libs/kotaemon/kotaemon/indices/ingests/files.py index f289567b7..2cb4a7178 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/files.py +++ b/libs/kotaemon/kotaemon/indices/ingests/files.py @@ -2,13 +2,13 @@ from typing import Type from decouple import config +from ktem.extensions.extensions import extension_manager from llama_index.core.readers.base import BaseReader from llama_index.readers.file import PDFReader from theflow.settings import settings as flowsettings from kotaemon.base import BaseComponent, Document, Param from kotaemon.indices.extractors import BaseDocParser -from kotaemon.indices.ingests.extensions import extension_manager from kotaemon.indices.splitters import BaseSplitter, TokenSplitter from kotaemon.loaders import ( AdobeReader, diff --git a/libs/ktem/ktem/app.py b/libs/ktem/ktem/app.py index 53ec58d80..81a68ed12 100644 --- a/libs/ktem/ktem/app.py +++ b/libs/ktem/ktem/app.py @@ -7,13 +7,12 @@ from ktem.assets import PDFJS_PREBUILT_DIR, KotaemonTheme from ktem.components import reasonings from ktem.exceptions import HookAlreadyDeclared, HookNotDeclared +from ktem.extensions.extensions import extension_manager from ktem.index import IndexManager from ktem.settings import BaseSettingGroup, SettingGroup, SettingReasoningGroup from theflow.settings import settings from theflow.utils.modules import import_dotted_string -from kotaemon.indices.ingests.extensions import extension_manager - class BaseApp: """The main app of Kotaemon diff --git a/libs/ktem/ktem/extensions/__init__.py b/libs/ktem/ktem/extensions/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/libs/kotaemon/kotaemon/indices/ingests/extensions.py b/libs/ktem/ktem/extensions/extensions.py similarity index 51% rename from libs/kotaemon/kotaemon/indices/ingests/extensions.py rename to libs/ktem/ktem/extensions/extensions.py index 0d956b96b..2cad24a9b 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/extensions.py +++ b/libs/ktem/ktem/extensions/extensions.py @@ -1,4 +1,5 @@ from copy import deepcopy +from functools import cached_property from typing import Any from decouple import config @@ -8,6 +9,7 @@ from kotaemon.loaders import ( AdobeReader, AzureAIDocumentIntelligenceLoader, + DoclingReader, GOCR2ImageReader, HtmlReader, MhtmlReader, @@ -15,24 +17,69 @@ PDFThumbnailReader, TxtReader, UnstructuredReader, + WebReader, ) -unstructured = UnstructuredReader() -adobe_reader = AdobeReader() -azure_reader = AzureAIDocumentIntelligenceLoader( - endpoint=str(config("AZURE_DI_ENDPOINT", default="")), - credential=str(config("AZURE_DI_CREDENTIAL", default="")), - cache_dir=getattr(flowsettings, "KH_MARKDOWN_OUTPUT_DIR", None), -) -adobe_reader.vlm_endpoint = azure_reader.vlm_endpoint = getattr( - flowsettings, "KH_VLM_ENDPOINT", "" -) + +class ReaderFactory: + @cached_property + def web(self) -> WebReader: + return WebReader() + + @cached_property + def unstructured(self) -> UnstructuredReader: + return UnstructuredReader() + + @cached_property + def adobe(self) -> AdobeReader: + adobe_reader = AdobeReader() + adobe_reader.vlm_endpoint = getattr(flowsettings, "KH_VLM_ENDPOINT", "") + return adobe_reader + + @cached_property + def azuredi(self) -> AzureAIDocumentIntelligenceLoader: + azuredi_reader = AzureAIDocumentIntelligenceLoader( + endpoint=str(config("AZURE_DI_ENDPOINT", default="")), + credential=str(config("AZURE_DI_CREDENTIAL", default="")), + cache_dir=getattr(flowsettings, "KH_MARKDOWN_OUTPUT_DIR", None), + ) + azuredi_reader.vlm_endpoint = getattr(flowsettings, "KH_VLM_ENDPOINT", "") + return azuredi_reader + + @cached_property + def pandas_excel(self) -> PandasExcelReader: + return PandasExcelReader() + + @cached_property + def html(self) -> HtmlReader: + return HtmlReader() + + @cached_property + def mhtml(self) -> MhtmlReader: + return MhtmlReader() + + @cached_property + def gocr(self) -> GOCR2ImageReader: + return GOCR2ImageReader() + + @cached_property + def txt(self) -> TxtReader: + return TxtReader() + + @cached_property + def docling(self) -> DoclingReader: + return DoclingReader() + + @cached_property + def pdf_thumbnail(self) -> PDFThumbnailReader: + return PDFThumbnailReader() class ExtensionManager: """Pool of loaders for extensions""" - def __init__(self): + def __init__(self, factory: ReaderFactory | None = None): + self.factory = factory or ReaderFactory() self._supported, self._default_index = self._init_supported() def get_current_loader(self) -> dict[str, BaseReader]: @@ -43,26 +90,40 @@ def get_current_loader(self) -> dict[str, BaseReader]: } ) - @staticmethod - def _init_supported() -> tuple[dict[str, list[BaseReader]], dict[str, str]]: - gocr = GOCR2ImageReader() - + def _init_supported(self) -> tuple[dict[str, list[BaseReader]], dict[str, str]]: supported: dict[str, list[BaseReader]] = { - ".xlsx": [PandasExcelReader()], - ".docx": [unstructured], - ".pptx": [unstructured], - ".xls": [unstructured], - ".doc": [unstructured], - ".html": [HtmlReader()], - ".mhtml": [MhtmlReader()], - ".png": [unstructured, gocr], - ".jpeg": [unstructured, gocr], - ".jpg": [unstructured, gocr], - ".tiff": [unstructured], - ".tif": [unstructured], - ".pdf": [PDFThumbnailReader(), adobe_reader, azure_reader], - ".txt": [TxtReader()], - ".md": [TxtReader()], + ".xlsx": [self.factory.pandas_excel], + ".docx": [self.factory.unstructured], + ".pptx": [self.factory.unstructured], + ".xls": [self.factory.unstructured], + ".doc": [self.factory.unstructured], + ".html": [self.factory.html], + ".mhtml": [self.factory.mhtml], + ".png": [ + self.factory.unstructured, + self.factory.gocr, + self.factory.docling, + ], + ".jpeg": [ + self.factory.unstructured, + self.factory.gocr, + self.factory.docling, + ], + ".jpg": [ + self.factory.unstructured, + self.factory.gocr, + self.factory.docling, + ], + ".tiff": [self.factory.unstructured, self.factory.docling], + ".tif": [self.factory.unstructured, self.factory.docling], + ".pdf": [ + self.factory.pdf_thumbnail, + self.factory.adobe, + self.factory.azuredi, + self.factory.docling, + ], + ".txt": [self.factory.txt], + ".md": [self.factory.txt], } default_index = { @@ -136,7 +197,3 @@ def generate_gradio_settings(self) -> dict[str, Any]: extension_manager = ExtensionManager() - - -if __name__ == "__main__": - print(extension_manager.get_loaders_by_extension(".xlsx")) diff --git a/libs/ktem/ktem/index/file/pipelines.py b/libs/ktem/ktem/index/file/pipelines.py index e74f1af0e..c6b555930 100644 --- a/libs/ktem/ktem/index/file/pipelines.py +++ b/libs/ktem/ktem/index/file/pipelines.py @@ -15,6 +15,7 @@ import tiktoken from ktem.db.models import engine from ktem.embeddings.manager import embedding_models_manager +from ktem.extensions.extensions import extension_manager from ktem.llms.manager import llms from ktem.rerankings.manager import reranking_models_manager from llama_index.core.readers.base import BaseReader @@ -34,14 +35,10 @@ from kotaemon.base import BaseComponent, Document, Node, Param, RetrievedDocument from kotaemon.embeddings import BaseEmbeddings from kotaemon.indices import VectorIndexing, VectorRetrieval -from kotaemon.indices.ingests.extensions import extension_manager -from kotaemon.indices.ingests.files import ( # KH_DEFAULT_FILE_EXTRACTORS, - adobe_reader, - azure_reader, - docling_reader, - unstructured, - web_reader, -) + +# from kotaemon.indices.ingests.files import ( # KH_DEFAULT_FILE_EXTRACTORS, +# web_reader, +# ) from kotaemon.indices.rankings import BaseReranking, LLMReranking, LLMTrulensScoring from kotaemon.indices.splitters import BaseSplitter, TokenSplitter @@ -674,11 +671,11 @@ def readers(self): readers: dict[str, BaseReader] = extension_manager.get_current_loader() print("reader_mode", self.reader_mode) if self.reader_mode == "adobe": - readers[".pdf"] = adobe_reader + readers[".pdf"] = extension_manager.factory.adobe elif self.reader_mode == "azure-di": - readers[".pdf"] = azure_reader + readers[".pdf"] = extension_manager.factory.azuredi elif self.reader_mode == "docling": - readers[".pdf"] = docling_reader + readers[".pdf"] = extension_manager.factory.docling dev_readers, _, _ = dev_settings() readers.update(dev_readers) @@ -737,11 +734,11 @@ def route(self, file_path: str | Path) -> IndexPipeline: # check if file_path is a URL if self.is_url(file_path): - reader = web_reader + reader = extension_manager.factory.web else: assert isinstance(file_path, Path) ext = file_path.suffix.lower() - reader = self.readers.get(ext, unstructured) + reader = self.readers.get(ext, extension_manager.factory.unstructured) if reader is None: raise NotImplementedError( f"No supported pipeline to index {file_path.name}. Please specify " diff --git a/libs/ktem/ktem/pages/chat/__init__.py b/libs/ktem/ktem/pages/chat/__init__.py index 970706591..b1f04de1f 100644 --- a/libs/ktem/ktem/pages/chat/__init__.py +++ b/libs/ktem/ktem/pages/chat/__init__.py @@ -8,6 +8,7 @@ from ktem.app import BasePage from ktem.components import reasonings from ktem.db.models import Conversation, engine +from ktem.extensions.extensions import extension_manager from ktem.index.file.ui import File from ktem.reasoning.prompt_optimization.suggest_conversation_name import ( SuggestConvNamePipeline, @@ -20,7 +21,6 @@ from theflow.settings import settings as flowsettings from kotaemon.base import Document -from kotaemon.indices.ingests.extensions import extension_manager from ...utils import SUPPORTED_LANGUAGE_MAP, get_file_names_regex from .chat_panel import ChatPanel diff --git a/libs/ktem/ktem/pages/settings.py b/libs/ktem/ktem/pages/settings.py index 001dcaef7..acbff117c 100644 --- a/libs/ktem/ktem/pages/settings.py +++ b/libs/ktem/ktem/pages/settings.py @@ -4,10 +4,9 @@ from ktem.app import BasePage from ktem.components import reasonings from ktem.db.models import Settings, User, engine +from ktem.extensions.extensions import extension_manager from sqlmodel import Session, select -from kotaemon.indices.ingests.extensions import extension_manager - signout_js = """ function(u, c, pw, pwc) { removeFromStorage('username'); From 9bbd8a17cb23e5e50a843268593733464cdde574 Mon Sep 17 00:00:00 2001 From: phv2312 Date: Sun, 15 Dec 2024 16:00:15 +0700 Subject: [PATCH 16/18] feat: bring extension manager to kotaemon --- .../kotaemon/indices/ingests}/extensions.py | 5 +++ .../kotaemon/indices/ingests/files.py | 43 ++++--------------- libs/ktem/ktem/app.py | 3 +- libs/ktem/ktem/extensions/__init__.py | 0 libs/ktem/ktem/index/file/pipelines.py | 2 +- libs/ktem/ktem/pages/chat/__init__.py | 2 +- libs/ktem/ktem/pages/settings.py | 3 +- 7 files changed, 20 insertions(+), 38 deletions(-) rename libs/{ktem/ktem/extensions => kotaemon/kotaemon/indices/ingests}/extensions.py (95%) delete mode 100644 libs/ktem/ktem/extensions/__init__.py diff --git a/libs/ktem/ktem/extensions/extensions.py b/libs/kotaemon/kotaemon/indices/ingests/extensions.py similarity index 95% rename from libs/ktem/ktem/extensions/extensions.py rename to libs/kotaemon/kotaemon/indices/ingests/extensions.py index 2cad24a9b..458efb7d5 100644 --- a/libs/ktem/ktem/extensions/extensions.py +++ b/libs/kotaemon/kotaemon/indices/ingests/extensions.py @@ -12,6 +12,7 @@ DoclingReader, GOCR2ImageReader, HtmlReader, + MathpixPDFReader, MhtmlReader, PandasExcelReader, PDFThumbnailReader, @@ -22,6 +23,10 @@ class ReaderFactory: + @cached_property + def mathpix_pdf(self) -> MathpixPDFReader: + return MathpixPDFReader() + @cached_property def web(self) -> WebReader: return WebReader() diff --git a/libs/kotaemon/kotaemon/indices/ingests/files.py b/libs/kotaemon/kotaemon/indices/ingests/files.py index 2cb4a7178..0ad9d3f31 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/files.py +++ b/libs/kotaemon/kotaemon/indices/ingests/files.py @@ -1,38 +1,14 @@ from pathlib import Path from typing import Type -from decouple import config -from ktem.extensions.extensions import extension_manager from llama_index.core.readers.base import BaseReader from llama_index.readers.file import PDFReader -from theflow.settings import settings as flowsettings from kotaemon.base import BaseComponent, Document, Param from kotaemon.indices.extractors import BaseDocParser +from kotaemon.indices.ingests.extensions import extension_manager from kotaemon.indices.splitters import BaseSplitter, TokenSplitter -from kotaemon.loaders import ( - AdobeReader, - AzureAIDocumentIntelligenceLoader, - DirectoryReader, - DoclingReader, - MathpixPDFReader, - OCRReader, - UnstructuredReader, - WebReader, -) - -web_reader = WebReader() -unstructured = UnstructuredReader() -adobe_reader = AdobeReader() -azure_reader = AzureAIDocumentIntelligenceLoader( - endpoint=str(config("AZURE_DI_ENDPOINT", default="")), - credential=str(config("AZURE_DI_CREDENTIAL", default="")), - cache_dir=getattr(flowsettings, "KH_MARKDOWN_OUTPUT_DIR", None), -) -docling_reader = DoclingReader() -adobe_reader.vlm_endpoint = ( - azure_reader.vlm_endpoint -) = docling_reader.vlm_endpoint = getattr(flowsettings, "KH_VLM_ENDPOINT", "") +from kotaemon.loaders import DirectoryReader class DocumentIngestor(BaseComponent): @@ -73,14 +49,13 @@ def _get_reader(self, input_files: list[str | Path]): for ext, cls in self.override_file_extractors.items(): file_extractors[ext] = cls() - if self.pdf_mode == "normal": - file_extractors[".pdf"] = PDFReader() - elif self.pdf_mode == "ocr": - file_extractors[".pdf"] = OCRReader() - elif self.pdf_mode == "multimodal": - file_extractors[".pdf"] = AdobeReader() - else: - file_extractors[".pdf"] = MathpixPDFReader() + match self.pdf_mode: + case "normal": + file_extractors[".pdf"] = PDFReader() + case "multimodal": + file_extractors[".pdf"] = extension_manager.factory.adobe + case _: + file_extractors[".pdf"] = extension_manager.factory.mathpix_pdf main_reader = DirectoryReader( input_files=input_files, diff --git a/libs/ktem/ktem/app.py b/libs/ktem/ktem/app.py index 81a68ed12..53ec58d80 100644 --- a/libs/ktem/ktem/app.py +++ b/libs/ktem/ktem/app.py @@ -7,12 +7,13 @@ from ktem.assets import PDFJS_PREBUILT_DIR, KotaemonTheme from ktem.components import reasonings from ktem.exceptions import HookAlreadyDeclared, HookNotDeclared -from ktem.extensions.extensions import extension_manager from ktem.index import IndexManager from ktem.settings import BaseSettingGroup, SettingGroup, SettingReasoningGroup from theflow.settings import settings from theflow.utils.modules import import_dotted_string +from kotaemon.indices.ingests.extensions import extension_manager + class BaseApp: """The main app of Kotaemon diff --git a/libs/ktem/ktem/extensions/__init__.py b/libs/ktem/ktem/extensions/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/libs/ktem/ktem/index/file/pipelines.py b/libs/ktem/ktem/index/file/pipelines.py index c6b555930..fc06eeae5 100644 --- a/libs/ktem/ktem/index/file/pipelines.py +++ b/libs/ktem/ktem/index/file/pipelines.py @@ -15,7 +15,6 @@ import tiktoken from ktem.db.models import engine from ktem.embeddings.manager import embedding_models_manager -from ktem.extensions.extensions import extension_manager from ktem.llms.manager import llms from ktem.rerankings.manager import reranking_models_manager from llama_index.core.readers.base import BaseReader @@ -35,6 +34,7 @@ from kotaemon.base import BaseComponent, Document, Node, Param, RetrievedDocument from kotaemon.embeddings import BaseEmbeddings from kotaemon.indices import VectorIndexing, VectorRetrieval +from kotaemon.indices.ingests.extensions import extension_manager # from kotaemon.indices.ingests.files import ( # KH_DEFAULT_FILE_EXTRACTORS, # web_reader, diff --git a/libs/ktem/ktem/pages/chat/__init__.py b/libs/ktem/ktem/pages/chat/__init__.py index b1f04de1f..970706591 100644 --- a/libs/ktem/ktem/pages/chat/__init__.py +++ b/libs/ktem/ktem/pages/chat/__init__.py @@ -8,7 +8,6 @@ from ktem.app import BasePage from ktem.components import reasonings from ktem.db.models import Conversation, engine -from ktem.extensions.extensions import extension_manager from ktem.index.file.ui import File from ktem.reasoning.prompt_optimization.suggest_conversation_name import ( SuggestConvNamePipeline, @@ -21,6 +20,7 @@ from theflow.settings import settings as flowsettings from kotaemon.base import Document +from kotaemon.indices.ingests.extensions import extension_manager from ...utils import SUPPORTED_LANGUAGE_MAP, get_file_names_regex from .chat_panel import ChatPanel diff --git a/libs/ktem/ktem/pages/settings.py b/libs/ktem/ktem/pages/settings.py index acbff117c..001dcaef7 100644 --- a/libs/ktem/ktem/pages/settings.py +++ b/libs/ktem/ktem/pages/settings.py @@ -4,9 +4,10 @@ from ktem.app import BasePage from ktem.components import reasonings from ktem.db.models import Settings, User, engine -from ktem.extensions.extensions import extension_manager from sqlmodel import Session, select +from kotaemon.indices.ingests.extensions import extension_manager + signout_js = """ function(u, c, pw, pwc) { removeFromStorage('username'); From 038dabbc61260a399702d830dadf2e14fb18a0e3 Mon Sep 17 00:00:00 2001 From: cin-niko Date: Mon, 16 Dec 2024 05:52:44 +0000 Subject: [PATCH 17/18] refactor: move exteions manager to ktem --- .../kotaemon/indices/ingests/__init__.py | 3 - .../kotaemon/indices/ingests/files.py | 90 ----- libs/kotaemon/tests/test_ingestor.py | 15 - libs/ktem/ktem/app.py | 3 +- libs/ktem/ktem/index/file/pipelines.py | 6 +- .../ktem/loaders}/extensions.py | 334 +++++++----------- libs/ktem/ktem/loaders/factory.py | 77 ++++ libs/ktem/ktem/pages/chat/__init__.py | 5 +- libs/ktem/ktem/pages/settings.py | 3 +- 9 files changed, 211 insertions(+), 325 deletions(-) delete mode 100644 libs/kotaemon/kotaemon/indices/ingests/__init__.py delete mode 100644 libs/kotaemon/kotaemon/indices/ingests/files.py delete mode 100644 libs/kotaemon/tests/test_ingestor.py rename libs/{kotaemon/kotaemon/indices/ingests => ktem/ktem/loaders}/extensions.py (66%) create mode 100644 libs/ktem/ktem/loaders/factory.py diff --git a/libs/kotaemon/kotaemon/indices/ingests/__init__.py b/libs/kotaemon/kotaemon/indices/ingests/__init__.py deleted file mode 100644 index 064f20623..000000000 --- a/libs/kotaemon/kotaemon/indices/ingests/__init__.py +++ /dev/null @@ -1,3 +0,0 @@ -from .files import DocumentIngestor - -__all__ = ["DocumentIngestor"] diff --git a/libs/kotaemon/kotaemon/indices/ingests/files.py b/libs/kotaemon/kotaemon/indices/ingests/files.py deleted file mode 100644 index 0ad9d3f31..000000000 --- a/libs/kotaemon/kotaemon/indices/ingests/files.py +++ /dev/null @@ -1,90 +0,0 @@ -from pathlib import Path -from typing import Type - -from llama_index.core.readers.base import BaseReader -from llama_index.readers.file import PDFReader - -from kotaemon.base import BaseComponent, Document, Param -from kotaemon.indices.extractors import BaseDocParser -from kotaemon.indices.ingests.extensions import extension_manager -from kotaemon.indices.splitters import BaseSplitter, TokenSplitter -from kotaemon.loaders import DirectoryReader - - -class DocumentIngestor(BaseComponent): - """Ingest common office document types into Document for indexing - - Document types: - - pdf - - xlsx, xls - - docx, doc - - Args: - pdf_mode: mode for pdf extraction, one of "normal", "mathpix", "ocr" - - normal: parse pdf text - - mathpix: parse pdf text using mathpix - - ocr: parse pdf image using flax - doc_parsers: list of document parsers to parse the document - text_splitter: splitter to split the document into text nodes - override_file_extractors: override file extractors for specific file extensions - The default file extractors are stored in `KH_DEFAULT_FILE_EXTRACTORS` - """ - - pdf_mode: str = "normal" # "normal", "mathpix", "ocr", "multimodal" - doc_parsers: list[BaseDocParser] = Param(default_callback=lambda _: []) - text_splitter: BaseSplitter = TokenSplitter.withx( - chunk_size=1024, - chunk_overlap=256, - separator="\n\n", - backup_separators=["\n", ".", " ", "\u200B"], - ) - override_file_extractors: dict[str, Type[BaseReader]] = {} - - def _get_reader(self, input_files: list[str | Path]): - """Get appropriate readers for the input files based on file extension""" - file_extractors: dict[str, BaseReader] = { - ext: reader - for ext, reader in extension_manager.get_current_loader().items() - } - for ext, cls in self.override_file_extractors.items(): - file_extractors[ext] = cls() - - match self.pdf_mode: - case "normal": - file_extractors[".pdf"] = PDFReader() - case "multimodal": - file_extractors[".pdf"] = extension_manager.factory.adobe - case _: - file_extractors[".pdf"] = extension_manager.factory.mathpix_pdf - - main_reader = DirectoryReader( - input_files=input_files, - file_extractor=file_extractors, - ) - - return main_reader - - def run(self, file_paths: list[str | Path] | str | Path) -> list[Document]: - """Ingest the file paths into Document - - Args: - file_paths: list of file paths or a single file path - - Returns: - list of parsed Documents - """ - if not isinstance(file_paths, list): - file_paths = [file_paths] - - documents = self._get_reader(input_files=file_paths)() - print(f"Read {len(file_paths)} files into {len(documents)} documents.") - nodes = self.text_splitter(documents) - print(f"Transform {len(documents)} documents into {len(nodes)} nodes.") - self.log_progress(".num_docs", num_docs=len(nodes)) - - # document parsers call - if self.doc_parsers: - for parser in self.doc_parsers: - nodes = parser(nodes) - - return nodes diff --git a/libs/kotaemon/tests/test_ingestor.py b/libs/kotaemon/tests/test_ingestor.py deleted file mode 100644 index 33fa5a235..000000000 --- a/libs/kotaemon/tests/test_ingestor.py +++ /dev/null @@ -1,15 +0,0 @@ -from pathlib import Path - -from kotaemon.indices.ingests import DocumentIngestor -from kotaemon.indices.splitters import TokenSplitter - - -def test_ingestor_include_src(): - dirpath = Path(__file__).parent - ingestor = DocumentIngestor( - pdf_mode="normal", - text_splitter=TokenSplitter(chunk_size=200, chunk_overlap=10), - ) - nodes = ingestor(dirpath / "resources" / "table.pdf") - assert type(nodes) is list - assert nodes[0].relationships diff --git a/libs/ktem/ktem/app.py b/libs/ktem/ktem/app.py index 53ec58d80..4734372c9 100644 --- a/libs/ktem/ktem/app.py +++ b/libs/ktem/ktem/app.py @@ -8,12 +8,11 @@ from ktem.components import reasonings from ktem.exceptions import HookAlreadyDeclared, HookNotDeclared from ktem.index import IndexManager +from ktem.loaders.extensions import extension_manager from ktem.settings import BaseSettingGroup, SettingGroup, SettingReasoningGroup from theflow.settings import settings from theflow.utils.modules import import_dotted_string -from kotaemon.indices.ingests.extensions import extension_manager - class BaseApp: """The main app of Kotaemon diff --git a/libs/ktem/ktem/index/file/pipelines.py b/libs/ktem/ktem/index/file/pipelines.py index fc06eeae5..6de40e86e 100644 --- a/libs/ktem/ktem/index/file/pipelines.py +++ b/libs/ktem/ktem/index/file/pipelines.py @@ -16,6 +16,7 @@ from ktem.db.models import engine from ktem.embeddings.manager import embedding_models_manager from ktem.llms.manager import llms +from ktem.loaders.extensions import extension_manager from ktem.rerankings.manager import reranking_models_manager from llama_index.core.readers.base import BaseReader from llama_index.core.readers.file.base import default_file_metadata_func @@ -34,11 +35,6 @@ from kotaemon.base import BaseComponent, Document, Node, Param, RetrievedDocument from kotaemon.embeddings import BaseEmbeddings from kotaemon.indices import VectorIndexing, VectorRetrieval -from kotaemon.indices.ingests.extensions import extension_manager - -# from kotaemon.indices.ingests.files import ( # KH_DEFAULT_FILE_EXTRACTORS, -# web_reader, -# ) from kotaemon.indices.rankings import BaseReranking, LLMReranking, LLMTrulensScoring from kotaemon.indices.splitters import BaseSplitter, TokenSplitter diff --git a/libs/kotaemon/kotaemon/indices/ingests/extensions.py b/libs/ktem/ktem/loaders/extensions.py similarity index 66% rename from libs/kotaemon/kotaemon/indices/ingests/extensions.py rename to libs/ktem/ktem/loaders/extensions.py index 458efb7d5..d05f8255b 100644 --- a/libs/kotaemon/kotaemon/indices/ingests/extensions.py +++ b/libs/ktem/ktem/loaders/extensions.py @@ -1,204 +1,130 @@ -from copy import deepcopy -from functools import cached_property -from typing import Any - -from decouple import config -from llama_index.core.readers.base import BaseReader -from theflow.settings import settings as flowsettings - -from kotaemon.loaders import ( - AdobeReader, - AzureAIDocumentIntelligenceLoader, - DoclingReader, - GOCR2ImageReader, - HtmlReader, - MathpixPDFReader, - MhtmlReader, - PandasExcelReader, - PDFThumbnailReader, - TxtReader, - UnstructuredReader, - WebReader, -) - - -class ReaderFactory: - @cached_property - def mathpix_pdf(self) -> MathpixPDFReader: - return MathpixPDFReader() - - @cached_property - def web(self) -> WebReader: - return WebReader() - - @cached_property - def unstructured(self) -> UnstructuredReader: - return UnstructuredReader() - - @cached_property - def adobe(self) -> AdobeReader: - adobe_reader = AdobeReader() - adobe_reader.vlm_endpoint = getattr(flowsettings, "KH_VLM_ENDPOINT", "") - return adobe_reader - - @cached_property - def azuredi(self) -> AzureAIDocumentIntelligenceLoader: - azuredi_reader = AzureAIDocumentIntelligenceLoader( - endpoint=str(config("AZURE_DI_ENDPOINT", default="")), - credential=str(config("AZURE_DI_CREDENTIAL", default="")), - cache_dir=getattr(flowsettings, "KH_MARKDOWN_OUTPUT_DIR", None), - ) - azuredi_reader.vlm_endpoint = getattr(flowsettings, "KH_VLM_ENDPOINT", "") - return azuredi_reader - - @cached_property - def pandas_excel(self) -> PandasExcelReader: - return PandasExcelReader() - - @cached_property - def html(self) -> HtmlReader: - return HtmlReader() - - @cached_property - def mhtml(self) -> MhtmlReader: - return MhtmlReader() - - @cached_property - def gocr(self) -> GOCR2ImageReader: - return GOCR2ImageReader() - - @cached_property - def txt(self) -> TxtReader: - return TxtReader() - - @cached_property - def docling(self) -> DoclingReader: - return DoclingReader() - - @cached_property - def pdf_thumbnail(self) -> PDFThumbnailReader: - return PDFThumbnailReader() - - -class ExtensionManager: - """Pool of loaders for extensions""" - - def __init__(self, factory: ReaderFactory | None = None): - self.factory = factory or ReaderFactory() - self._supported, self._default_index = self._init_supported() - - def get_current_loader(self) -> dict[str, BaseReader]: - return deepcopy( - { - k: self.get_selected_loader_by_extension(k)[0] - for k, _ in self._supported.items() - } - ) - - def _init_supported(self) -> tuple[dict[str, list[BaseReader]], dict[str, str]]: - supported: dict[str, list[BaseReader]] = { - ".xlsx": [self.factory.pandas_excel], - ".docx": [self.factory.unstructured], - ".pptx": [self.factory.unstructured], - ".xls": [self.factory.unstructured], - ".doc": [self.factory.unstructured], - ".html": [self.factory.html], - ".mhtml": [self.factory.mhtml], - ".png": [ - self.factory.unstructured, - self.factory.gocr, - self.factory.docling, - ], - ".jpeg": [ - self.factory.unstructured, - self.factory.gocr, - self.factory.docling, - ], - ".jpg": [ - self.factory.unstructured, - self.factory.gocr, - self.factory.docling, - ], - ".tiff": [self.factory.unstructured, self.factory.docling], - ".tif": [self.factory.unstructured, self.factory.docling], - ".pdf": [ - self.factory.pdf_thumbnail, - self.factory.adobe, - self.factory.azuredi, - self.factory.docling, - ], - ".txt": [self.factory.txt], - ".md": [self.factory.txt], - } - - default_index = { - k: ExtensionManager.get_loader_name(vs[0]) for k, vs in supported.items() - } - - return supported, default_index - - def load(self, settings: dict, prefix="extension"): - for key, value in settings.items(): - if not key.startswith(prefix): - continue - extension = key.replace("extension.", "") - if extension in self._supported: - # Update the default index - # Only if it's in supported list - supported_loader_names = self.get_loaders_by_extension(extension)[1] - if value in supported_loader_names: - self._default_index[extension] = value - else: - print( - f"[{extension}]Can not find loader: {value} from list of " - f"supported extensions: {supported_loader_names}" - ) - - @staticmethod - def get_loader_name(loader: BaseReader) -> str: - return loader.__class__.__name__ - - def get_supported_extensions(self): - return list(self._supported.keys()) - - def get_loaders_by_extension( - self, extension: str - ) -> tuple[list[BaseReader], list[str]]: - loaders = self._supported[extension] - loaders_name = [self.get_loader_name(loader) for loader in loaders] - return loaders, loaders_name - - def get_selected_loader_by_extension( - self, extension: str - ) -> tuple[BaseReader, str]: - supported_loaders: list[BaseReader] = self._supported[extension] - - for loader in supported_loaders: - loader_name = self.get_loader_name(loader) - - if loader_name == self._default_index[extension]: - return loader, loader_name - - raise Exception(f"can not find the selected loader for extension: {extension}") - - def generate_gradio_settings(self) -> dict[str, Any]: - """Generates the settings dictionary for use in Gradio.""" - settings = {} - - for extension, loaders in self._supported.items(): - current_loader: str = self._default_index[extension] - loaders_choices: list[str] = [ - self.get_loader_name(loader) for loader in loaders - ] - - settings[extension] = { - "name": f"Loader {extension}", - "value": current_loader, - "choices": loaders_choices, - "component": "dropdown", # You can customize this to "radio" if needed - } - - return settings - - -extension_manager = ExtensionManager() +from copy import deepcopy +from typing import Any + +from llama_index.core.readers.base import BaseReader + +from .factory import ReaderFactory + + +class ExtensionManager: + """Pool of loaders for extensions""" + + def __init__(self, factory: ReaderFactory | None = None): + self.factory = factory or ReaderFactory() + self._supported, self._default_index = self._init_supported() + + def get_current_loader(self) -> dict[str, BaseReader]: + return deepcopy( + { + k: self.get_selected_loader_by_extension(k)[0] + for k, _ in self._supported.items() + } + ) + + def _init_supported(self) -> tuple[dict[str, list[BaseReader]], dict[str, str]]: + supported: dict[str, list[BaseReader]] = { + ".xlsx": [self.factory.pandas_excel], + ".docx": [self.factory.unstructured], + ".pptx": [self.factory.unstructured], + ".xls": [self.factory.unstructured], + ".doc": [self.factory.unstructured], + ".html": [self.factory.html], + ".mhtml": [self.factory.mhtml], + ".png": [ + self.factory.unstructured, + self.factory.gocr, + self.factory.docling, + ], + ".jpeg": [ + self.factory.unstructured, + self.factory.gocr, + self.factory.docling, + ], + ".jpg": [ + self.factory.unstructured, + self.factory.gocr, + self.factory.docling, + ], + ".tiff": [self.factory.unstructured, self.factory.docling], + ".tif": [self.factory.unstructured, self.factory.docling], + ".pdf": [ + self.factory.pdf_thumbnail, + self.factory.adobe, + self.factory.azuredi, + self.factory.docling, + ], + ".txt": [self.factory.txt], + ".md": [self.factory.txt], + } + + default_index = { + k: ExtensionManager.get_loader_name(vs[0]) for k, vs in supported.items() + } + + return supported, default_index + + def load(self, settings: dict, prefix="extension"): + for key, value in settings.items(): + if not key.startswith(prefix): + continue + extension = key.replace("extension.", "") + if extension in self._supported: + # Update the default index + # Only if it's in supported list + supported_loader_names = self.get_loaders_by_extension(extension)[1] + if value in supported_loader_names: + self._default_index[extension] = value + else: + print( + f"[{extension}]Can not find loader: {value} from list of " + f"supported extensions: {supported_loader_names}" + ) + + @staticmethod + def get_loader_name(loader: BaseReader) -> str: + return loader.__class__.__name__ + + def get_supported_extensions(self): + return list(self._supported.keys()) + + def get_loaders_by_extension( + self, extension: str + ) -> tuple[list[BaseReader], list[str]]: + loaders = self._supported[extension] + loaders_name = [self.get_loader_name(loader) for loader in loaders] + return loaders, loaders_name + + def get_selected_loader_by_extension( + self, extension: str + ) -> tuple[BaseReader, str]: + supported_loaders: list[BaseReader] = self._supported[extension] + + for loader in supported_loaders: + loader_name = self.get_loader_name(loader) + + if loader_name == self._default_index[extension]: + return loader, loader_name + + raise Exception(f"can not find the selected loader for extension: {extension}") + + def generate_gradio_settings(self) -> dict[str, Any]: + """Generates the settings dictionary for use in Gradio.""" + settings = {} + + for extension, loaders in self._supported.items(): + current_loader: str = self._default_index[extension] + loaders_choices: list[str] = [ + self.get_loader_name(loader) for loader in loaders + ] + + settings[extension] = { + "name": f"Loader {extension}", + "value": current_loader, + "choices": loaders_choices, + "component": "dropdown", # You can customize this to "radio" if needed + } + + return settings + + +extension_manager = ExtensionManager() diff --git a/libs/ktem/ktem/loaders/factory.py b/libs/ktem/ktem/loaders/factory.py new file mode 100644 index 000000000..477338adc --- /dev/null +++ b/libs/ktem/ktem/loaders/factory.py @@ -0,0 +1,77 @@ +from functools import cached_property + +from decouple import config +from theflow.settings import settings as flowsettings + +from kotaemon.loaders import ( + AdobeReader, + AzureAIDocumentIntelligenceLoader, + DoclingReader, + GOCR2ImageReader, + HtmlReader, + MathpixPDFReader, + MhtmlReader, + PandasExcelReader, + PDFThumbnailReader, + TxtReader, + UnstructuredReader, + WebReader, +) + + +class ReaderFactory: + @cached_property + def mathpix_pdf(self) -> MathpixPDFReader: + return MathpixPDFReader() + + @cached_property + def web(self) -> WebReader: + return WebReader() + + @cached_property + def unstructured(self) -> UnstructuredReader: + return UnstructuredReader() + + @cached_property + def adobe(self) -> AdobeReader: + adobe_reader = AdobeReader() + adobe_reader.vlm_endpoint = getattr(flowsettings, "KH_VLM_ENDPOINT", "") + return adobe_reader + + @cached_property + def azuredi(self) -> AzureAIDocumentIntelligenceLoader: + azuredi_reader = AzureAIDocumentIntelligenceLoader( + endpoint=str(config("AZURE_DI_ENDPOINT", default="")), + credential=str(config("AZURE_DI_CREDENTIAL", default="")), + cache_dir=getattr(flowsettings, "KH_MARKDOWN_OUTPUT_DIR", None), + ) + azuredi_reader.vlm_endpoint = getattr(flowsettings, "KH_VLM_ENDPOINT", "") + return azuredi_reader + + @cached_property + def pandas_excel(self) -> PandasExcelReader: + return PandasExcelReader() + + @cached_property + def html(self) -> HtmlReader: + return HtmlReader() + + @cached_property + def mhtml(self) -> MhtmlReader: + return MhtmlReader() + + @cached_property + def gocr(self) -> GOCR2ImageReader: + return GOCR2ImageReader() + + @cached_property + def txt(self) -> TxtReader: + return TxtReader() + + @cached_property + def docling(self) -> DoclingReader: + return DoclingReader() + + @cached_property + def pdf_thumbnail(self) -> PDFThumbnailReader: + return PDFThumbnailReader() diff --git a/libs/ktem/ktem/pages/chat/__init__.py b/libs/ktem/ktem/pages/chat/__init__.py index 970706591..1bdf8269e 100644 --- a/libs/ktem/ktem/pages/chat/__init__.py +++ b/libs/ktem/ktem/pages/chat/__init__.py @@ -9,6 +9,7 @@ from ktem.components import reasonings from ktem.db.models import Conversation, engine from ktem.index.file.ui import File +from ktem.loaders.extensions import extension_manager from ktem.reasoning.prompt_optimization.suggest_conversation_name import ( SuggestConvNamePipeline, ) @@ -20,7 +21,6 @@ from theflow.settings import settings as flowsettings from kotaemon.base import Document -from kotaemon.indices.ingests.extensions import extension_manager from ...utils import SUPPORTED_LANGUAGE_MAP, get_file_names_regex from .chat_panel import ChatPanel @@ -28,9 +28,6 @@ from .control import ConversationControl from .report import ReportIssue -# from kotaemon.indices.ingests.files import KH_DEFAULT_FILE_EXTRACTORS - - DEFAULT_SETTING = "(default)" INFO_PANEL_SCALES = {True: 8, False: 4} diff --git a/libs/ktem/ktem/pages/settings.py b/libs/ktem/ktem/pages/settings.py index 001dcaef7..fda68f54f 100644 --- a/libs/ktem/ktem/pages/settings.py +++ b/libs/ktem/ktem/pages/settings.py @@ -4,10 +4,9 @@ from ktem.app import BasePage from ktem.components import reasonings from ktem.db.models import Settings, User, engine +from ktem.loaders.extensions import extension_manager from sqlmodel import Session, select -from kotaemon.indices.ingests.extensions import extension_manager - signout_js = """ function(u, c, pw, pwc) { removeFromStorage('username'); From cb4fabcc84dbcf5b5b216d4fedce23c7a13b036d Mon Sep 17 00:00:00 2001 From: cin-niko Date: Mon, 16 Dec 2024 07:05:15 +0000 Subject: [PATCH 18/18] fix: remove duplicate loader setting in retrieval settings --- integration/got-ocr2.md | 2 +- libs/ktem/ktem/index/file/pipelines.py | 32 +------------------------- libs/ktem/ktem/pages/settings.py | 2 +- 3 files changed, 3 insertions(+), 33 deletions(-) diff --git a/integration/got-ocr2.md b/integration/got-ocr2.md index f569ddb4b..999cca714 100644 --- a/integration/got-ocr2.md +++ b/integration/got-ocr2.md @@ -26,5 +26,5 @@ docker run -d --gpus all -p 8881:8881 ghcr.io/phv2312/got-ocr2.0:main - Users can switch between multiple loaders for the same file extension, such as using the GOCR2ImageReader or a different unstructured data parser for .png files. This provides the flexibility to choose the best-suited loader for the task at hand. -- To change the default loader, go to **Settings**, then **Extension settings**. It displays a grid of extensions and +- To change the default loader, go to **Settings**, then **Loader settings**. It displays a grid of extensions and its supported loaders. Any modification will be saved to DB as other settings do. diff --git a/libs/ktem/ktem/index/file/pipelines.py b/libs/ktem/ktem/index/file/pipelines.py index 6de40e86e..651fabdd8 100644 --- a/libs/ktem/ktem/index/file/pipelines.py +++ b/libs/ktem/ktem/index/file/pipelines.py @@ -657,46 +657,17 @@ class IndexDocumentPipeline(BaseFileIndexIndexing): decide which pipeline should be used. """ - reader_mode: str = Param("default", help="The reader mode") embedding: BaseEmbeddings run_embedding_in_thread: bool = False - @Param.auto(depends_on="reader_mode") + @Param.auto() def readers(self): - # readers = deepcopy(KH_DEFAULT_FILE_EXTRACTORS) readers: dict[str, BaseReader] = extension_manager.get_current_loader() - print("reader_mode", self.reader_mode) - if self.reader_mode == "adobe": - readers[".pdf"] = extension_manager.factory.adobe - elif self.reader_mode == "azure-di": - readers[".pdf"] = extension_manager.factory.azuredi - elif self.reader_mode == "docling": - readers[".pdf"] = extension_manager.factory.docling - dev_readers, _, _ = dev_settings() readers.update(dev_readers) return readers - @classmethod - def get_user_settings(cls): - return { - "reader_mode": { - "name": "File loader", - "value": "default", - "choices": [ - ("Default (open-source)", "default"), - ("Adobe API (figure+table extraction)", "adobe"), - ( - "Azure AI Document Intelligence (figure+table extraction)", - "azure-di", - ), - ("Docling (figure+table extraction)", "docling"), - ], - "component": "dropdown", - }, - } - @classmethod def get_pipeline(cls, user_settings, index_settings) -> BaseFileIndexIndexing: use_quick_index_mode = user_settings.get("quick_index_mode", False) @@ -708,7 +679,6 @@ def get_pipeline(cls, user_settings, index_settings) -> BaseFileIndexIndexing: ) ], run_embedding_in_thread=use_quick_index_mode, - reader_mode=user_settings.get("reader_mode", "default"), ) return obj diff --git a/libs/ktem/ktem/pages/settings.py b/libs/ktem/ktem/pages/settings.py index fda68f54f..1f8a0e595 100644 --- a/libs/ktem/ktem/pages/settings.py +++ b/libs/ktem/ktem/pages/settings.py @@ -300,7 +300,7 @@ def extension_tab(self): assert len(lefts) == len(rights) - with gr.Tab("Extension settings"): + with gr.Tab("Loader settings"): for left, right in zip(lefts, rights): left_setting = self._default_settings.extension.settings.get(left, None) right_setting = self._default_settings.extension.settings.get(