mirror of
https://github.com/hwchase17/langchain
synced 2024-11-06 03:20:49 +00:00
b0b1a67771
### Description: When attempting to download PDF files from arXiv, an unexpected 404 error frequently occurs. This error halts the operation, regardless of whether there are additional documents to process. As a solution, I suggest implementing a mechanism to ignore and communicate this error and continue processing the next document from the list. Proposed Solution: To address the issue of unexpected 404 errors during PDF downloads from arXiv, I propose implementing the following solution: - Error Handling: Implement error handling mechanisms to catch and handle 404 errors gracefully. - Communication: Inform the user or logging system about the occurrence of the 404 error. - Continued Processing: After encountering a 404 error, continue processing the remaining documents from the list without interruption. This solution ensures that the application can handle unexpected errors without terminating the entire operation. It promotes resilience and robustness in the face of intermittent issues encountered during PDF downloads from arXiv. ### Issue: #20909 ### Dependencies: none --------- Co-authored-by: Bagatur <22008038+baskaryan@users.noreply.github.com>
260 lines
9.8 KiB
Python
260 lines
9.8 KiB
Python
"""Util that calls Arxiv."""
|
|
import logging
|
|
import os
|
|
import re
|
|
from typing import Any, Dict, Iterator, List, Optional
|
|
|
|
from langchain_core.documents import Document
|
|
from langchain_core.pydantic_v1 import BaseModel, root_validator
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
class ArxivAPIWrapper(BaseModel):
|
|
"""Wrapper around ArxivAPI.
|
|
|
|
To use, you should have the ``arxiv`` python package installed.
|
|
https://lukasschwab.me/arxiv.py/index.html
|
|
This wrapper will use the Arxiv API to conduct searches and
|
|
fetch document summaries. By default, it will return the document summaries
|
|
of the top-k results.
|
|
If the query is in the form of arxiv identifier
|
|
(see https://info.arxiv.org/help/find/index.html), it will return the paper
|
|
corresponding to the arxiv identifier.
|
|
It limits the Document content by doc_content_chars_max.
|
|
Set doc_content_chars_max=None if you don't want to limit the content size.
|
|
|
|
Attributes:
|
|
top_k_results: number of the top-scored document used for the arxiv tool
|
|
ARXIV_MAX_QUERY_LENGTH: the cut limit on the query used for the arxiv tool.
|
|
continue_on_failure (bool): If True, continue loading other URLs on failure.
|
|
load_max_docs: a limit to the number of loaded documents
|
|
load_all_available_meta:
|
|
if True: the `metadata` of the loaded Documents contains all available
|
|
meta info (see https://lukasschwab.me/arxiv.py/index.html#Result),
|
|
if False: the `metadata` contains only the published date, title,
|
|
authors and summary.
|
|
doc_content_chars_max: an optional cut limit for the length of a document's
|
|
content
|
|
|
|
Example:
|
|
.. code-block:: python
|
|
|
|
from langchain_community.utilities.arxiv import ArxivAPIWrapper
|
|
arxiv = ArxivAPIWrapper(
|
|
top_k_results = 3,
|
|
ARXIV_MAX_QUERY_LENGTH = 300,
|
|
load_max_docs = 3,
|
|
load_all_available_meta = False,
|
|
doc_content_chars_max = 40000
|
|
)
|
|
arxiv.run("tree of thought llm)
|
|
"""
|
|
|
|
arxiv_search: Any #: :meta private:
|
|
arxiv_exceptions: Any # :meta private:
|
|
top_k_results: int = 3
|
|
ARXIV_MAX_QUERY_LENGTH: int = 300
|
|
continue_on_failure: bool = False
|
|
load_max_docs: int = 100
|
|
load_all_available_meta: bool = False
|
|
doc_content_chars_max: Optional[int] = 4000
|
|
|
|
def is_arxiv_identifier(self, query: str) -> bool:
|
|
"""Check if a query is an arxiv identifier."""
|
|
arxiv_identifier_pattern = r"\d{2}(0[1-9]|1[0-2])\.\d{4,5}(v\d+|)|\d{7}.*"
|
|
for query_item in query[: self.ARXIV_MAX_QUERY_LENGTH].split():
|
|
match_result = re.match(arxiv_identifier_pattern, query_item)
|
|
if not match_result:
|
|
return False
|
|
assert match_result is not None
|
|
if not match_result.group(0) == query_item:
|
|
return False
|
|
return True
|
|
|
|
@root_validator()
|
|
def validate_environment(cls, values: Dict) -> Dict:
|
|
"""Validate that the python package exists in environment."""
|
|
try:
|
|
import arxiv
|
|
|
|
values["arxiv_search"] = arxiv.Search
|
|
values["arxiv_exceptions"] = (
|
|
arxiv.ArxivError,
|
|
arxiv.UnexpectedEmptyPageError,
|
|
arxiv.HTTPError,
|
|
)
|
|
values["arxiv_result"] = arxiv.Result
|
|
except ImportError:
|
|
raise ImportError(
|
|
"Could not import arxiv python package. "
|
|
"Please install it with `pip install arxiv`."
|
|
)
|
|
return values
|
|
|
|
def get_summaries_as_docs(self, query: str) -> List[Document]:
|
|
"""
|
|
Performs an arxiv search and returns list of
|
|
documents, with summaries as the content.
|
|
|
|
If an error occurs or no documents found, error text
|
|
is returned instead. Wrapper for
|
|
https://lukasschwab.me/arxiv.py/index.html#Search
|
|
|
|
Args:
|
|
query: a plaintext search query
|
|
""" # noqa: E501
|
|
try:
|
|
if self.is_arxiv_identifier(query):
|
|
results = self.arxiv_search(
|
|
id_list=query.split(),
|
|
max_results=self.top_k_results,
|
|
).results()
|
|
else:
|
|
results = self.arxiv_search( # type: ignore
|
|
query[: self.ARXIV_MAX_QUERY_LENGTH], max_results=self.top_k_results
|
|
).results()
|
|
except self.arxiv_exceptions as ex:
|
|
return [Document(page_content=f"Arxiv exception: {ex}")]
|
|
docs = [
|
|
Document(
|
|
page_content=result.summary,
|
|
metadata={
|
|
"Entry ID": result.entry_id,
|
|
"Published": result.updated.date(),
|
|
"Title": result.title,
|
|
"Authors": ", ".join(a.name for a in result.authors),
|
|
},
|
|
)
|
|
for result in results
|
|
]
|
|
return docs
|
|
|
|
def run(self, query: str) -> str:
|
|
"""
|
|
Performs an arxiv search and A single string
|
|
with the publish date, title, authors, and summary
|
|
for each article separated by two newlines.
|
|
|
|
If an error occurs or no documents found, error text
|
|
is returned instead. Wrapper for
|
|
https://lukasschwab.me/arxiv.py/index.html#Search
|
|
|
|
Args:
|
|
query: a plaintext search query
|
|
""" # noqa: E501
|
|
try:
|
|
if self.is_arxiv_identifier(query):
|
|
results = self.arxiv_search(
|
|
id_list=query.split(),
|
|
max_results=self.top_k_results,
|
|
).results()
|
|
else:
|
|
results = self.arxiv_search( # type: ignore
|
|
query[: self.ARXIV_MAX_QUERY_LENGTH], max_results=self.top_k_results
|
|
).results()
|
|
except self.arxiv_exceptions as ex:
|
|
return f"Arxiv exception: {ex}"
|
|
docs = [
|
|
f"Published: {result.updated.date()}\n"
|
|
f"Title: {result.title}\n"
|
|
f"Authors: {', '.join(a.name for a in result.authors)}\n"
|
|
f"Summary: {result.summary}"
|
|
for result in results
|
|
]
|
|
if docs:
|
|
return "\n\n".join(docs)[: self.doc_content_chars_max]
|
|
else:
|
|
return "No good Arxiv Result was found"
|
|
|
|
def load(self, query: str) -> List[Document]:
|
|
"""
|
|
Run Arxiv search and get the article texts plus the article meta information.
|
|
See https://lukasschwab.me/arxiv.py/index.html#Search
|
|
|
|
Returns: a list of documents with the document.page_content in text format
|
|
|
|
Performs an arxiv search, downloads the top k results as PDFs, loads
|
|
them as Documents, and returns them in a List.
|
|
|
|
Args:
|
|
query: a plaintext search query
|
|
"""
|
|
return list(self.lazy_load(query))
|
|
|
|
def lazy_load(self, query: str) -> Iterator[Document]:
|
|
"""
|
|
Run Arxiv search and get the article texts plus the article meta information.
|
|
See https://lukasschwab.me/arxiv.py/index.html#Search
|
|
|
|
Returns: documents with the document.page_content in text format
|
|
|
|
Performs an arxiv search, downloads the top k results as PDFs, loads
|
|
them as Documents, and returns them.
|
|
|
|
Args:
|
|
query: a plaintext search query
|
|
"""
|
|
try:
|
|
import fitz
|
|
except ImportError:
|
|
raise ImportError(
|
|
"PyMuPDF package not found, please install it with "
|
|
"`pip install pymupdf`"
|
|
)
|
|
|
|
try:
|
|
# Remove the ":" and "-" from the query, as they can cause search problems
|
|
query = query.replace(":", "").replace("-", "")
|
|
if self.is_arxiv_identifier(query):
|
|
results = self.arxiv_search(
|
|
id_list=query[: self.ARXIV_MAX_QUERY_LENGTH].split(),
|
|
max_results=self.load_max_docs,
|
|
).results()
|
|
else:
|
|
results = self.arxiv_search( # type: ignore
|
|
query[: self.ARXIV_MAX_QUERY_LENGTH], max_results=self.load_max_docs
|
|
).results()
|
|
except self.arxiv_exceptions as ex:
|
|
logger.debug("Error on arxiv: %s", ex)
|
|
return
|
|
|
|
for result in results:
|
|
try:
|
|
doc_file_name: str = result.download_pdf()
|
|
with fitz.open(doc_file_name) as doc_file:
|
|
text: str = "".join(page.get_text() for page in doc_file)
|
|
except (FileNotFoundError, fitz.fitz.FileDataError) as f_ex:
|
|
logger.debug(f_ex)
|
|
continue
|
|
except Exception as e:
|
|
if self.continue_on_failure:
|
|
logger.error(e)
|
|
continue
|
|
else:
|
|
raise e
|
|
if self.load_all_available_meta:
|
|
extra_metadata = {
|
|
"entry_id": result.entry_id,
|
|
"published_first_time": str(result.published.date()),
|
|
"comment": result.comment,
|
|
"journal_ref": result.journal_ref,
|
|
"doi": result.doi,
|
|
"primary_category": result.primary_category,
|
|
"categories": result.categories,
|
|
"links": [link.href for link in result.links],
|
|
}
|
|
else:
|
|
extra_metadata = {}
|
|
metadata = {
|
|
"Published": str(result.updated.date()),
|
|
"Title": result.title,
|
|
"Authors": ", ".join(a.name for a in result.authors),
|
|
"Summary": result.summary,
|
|
**extra_metadata,
|
|
}
|
|
yield Document(
|
|
page_content=text[: self.doc_content_chars_max], metadata=metadata
|
|
)
|
|
os.remove(doc_file_name)
|