jojortz's picture
add exception handlers for GPT and Claude
3966ab6
raw
history blame
8.51 kB
import base64
import json
import os
import requests
import anthropic
import openai
from dotenv import load_dotenv
from pathlib import Path
from llama_parse import LlamaParse
from llama_index.core import SimpleDirectoryReader
from unstructured.partition.auto import partition
from preprocessors.preprocessor import PdfPreprocessor
from postprocessors.postprocessor import ClaudePostprocessor, GPTPostprocessor
load_dotenv()
class Model:
BASE_URL: str | None = None
API_KEY: str | None = None
MODEL: str | None = None
def __init_subclass__(cls) -> None:
"""Initialize subclass."""
super().__init_subclass__()
def __init__(self):
"""Init self"""
def extract(self, file_path: str) -> str:
"""Extract model.
Args:
file_path: path to file to extract
Returns:
str: output markdown
"""
raise NotImplementedError("Model extract method is not implemented")
class AnyParserModel(Model):
BASE_URL = "https://k7u1c342dc.execute-api.us-west-2.amazonaws.com/v1/extract"
API_KEY = os.getenv('ANYPARSER_RT_API_KEY')
def extract(self, file_path: str) -> str:
"""Extract data in real-time.
Args:
file_path (str): The path to the file to be parsed.
Returns:
str: The extracted data.
"""
file_extension = Path(file_path).suffix.lower().lstrip(".")
# Check if the file exists
if not Path(file_path).is_file():
return "Error: File does not exist", "File does not exist"
if file_extension in ["pdf", "docx"]:
# Encode the PDF file content in base64
with open(file_path, "rb") as file:
encoded_file = base64.b64encode(file.read()).decode("utf-8")
else:
return "Error: Unsupported file type", "Unsupported file type"
# Create the JSON payload
payload = {
"file_content": encoded_file,
"file_type": file_extension,
}
# Set the headers
headers = {
"Content-Type": "application/json",
"x-api-key": self.API_KEY,
}
# Send the POST request
response = requests.post(
self.BASE_URL, headers=headers, data=json.dumps(payload), timeout=30
)
# Check if the request was successful
if response.status_code == 200:
try:
response_data = response.json()
response_list = []
for text in response_data["markdown"]:
response_list.append(text)
markdown_text = "\n".join(response_list)
return markdown_text
except json.JSONDecodeError:
return "Error: Invalid JSON response", f"Response: {response.text}"
else:
return f"Error: {response.status_code}", f"Response: {response.text}"
class LlamaParseModel(Model):
BASE_URL = None
API_KEY = os.getenv('LLAMA_CLOUD_API_KEY')
def __init__(self):
"""Init."""
super().__init__()
if not self.API_KEY:
raise ValueError("The API key is required. Please set the LLAMA_CLOUD_API_KEY environment variable.")
def extract(self, file_path: str) -> str:
"""Extract data in real-time.
Args:
file_path (str): The path to the file to be parsed.
Returns:
str: The extracted data.
"""
try:
parser = LlamaParse(
result_type="markdown",
num_workers=4,
verbose=True,
language="en",
)
file_extractor = {".pdf": parser}
documents = SimpleDirectoryReader(input_files=[file_path], file_extractor=file_extractor).load_data()
markdown = "\n\n".join([doc.text for doc in documents])
return markdown
except Exception as e:
print(f"Error processing input: {str(e)}")
return f"Error processing with LlamaParse: {str(e)}"
class UnstructuredModel(Model):
BASE_URL = None
API_KEY = None
def __init__(self):
"""Init."""
super().__init__()
def extract(self, file_path: str) -> str:
"""Extract data in real-time.
Args:
file_path (str): The path to the file to be parsed.
Returns:
str: The extracted data.
"""
try:
elements = partition(file_path)
parsed_text = "\n".join(str(element) for element in elements)
markdown = parsed_text if parsed_text else "No content parsed"
return markdown
except Exception as e:
return f"Error processing UnstructuredModel: {str(e)}"
class GPTModel(Model):
BASE_URL = None
API_KEY = os.getenv("OPENAI_API_KEY")
MODEL = "gpt-4o-mini"
REQUIRES_OPENAI = True
def __init__(self):
"""Init."""
super().__init__()
if not self.API_KEY:
raise ValueError(
"The API key is required. Please set the OPENAI_API_KEY environment variable."
)
self._client = openai.OpenAI(api_key=self.API_KEY)
def extract(self, file_path: str) -> str:
"""Extract data in real-time.
Args:
file_path (str): The path to the file to be parsed.
Returns:
str: The extracted data.
"""
try:
pdf_preprocessor = PdfPreprocessor()
gpt_postprocessor = GPTPostprocessor()
file_contents = pdf_preprocessor.run(file_path)
contents = []
for content in file_contents:
contents.append(
{
"type": "image_url",
"image_url": {
"url": f"data:image/jpeg;base64,{content}",
},
})
messages = [
{
"role": "user",
"content": [
{"type": "text", "text": "Convert this image to markdown"},
*contents,
],
}
]
response = self._client.chat.completions.create(
model=self.MODEL,
messages=messages,
)
return gpt_postprocessor.run(response.choices[0].message.content)
except Exception as e:
print(f"Error processing input: {str(e)}")
return f"Error processing with GPTModel: {str(e)}"
class ClaudeModel(Model):
BASE_URL = "http://103.114.163.134:3000/v1/"
API_KEY = os.getenv("ANTHROPIC_API_KEY")
MODEL = "claude-3-5-sonnet-20240620"
REQUIRES_OPENAI = True
def __init__(self):
"""Init."""
super().__init__()
if not self.API_KEY:
raise ValueError(
"The API key is required. Please set the ANTHROPIC_API_KEY environment variable."
)
self._client = anthropic.Anthropic(
api_key=self.API_KEY,
)
def extract(self, file_path: str) -> str:
"""Extract data in real-time.
Args:
file_path (str): The path to the file to be parsed.
Returns:
str: The extracted data.
"""
try:
prompt = "Convert this image to markdown."
pdf_preprocessor = PdfPreprocessor()
claude_postprocessor = ClaudePostprocessor()
file_contents = pdf_preprocessor.run(file_path)
contents = []
for content in file_contents:
contents.append(
{
"type": "image",
"source": {
"type": "base64",
"media_type": "image/jpeg",
"data": content,
}
})
messages = [
{"role": "user", "content": [
{"type": "text", "text": prompt},
*contents,
]}
]
response = self._client.messages.create(
model="claude-3-5-sonnet-20240620", max_tokens=1024, messages=messages
)
print(response.content[0].text)
return claude_postprocessor.run(response.content[0].text)
except Exception as e:
return f"Error processing ClaudeModel: {str(e)}"