|
import logging |
|
from dataclasses import dataclass |
|
from typing import Optional |
|
|
|
import requests |
|
from open_webui.env import SRC_LOG_LEVELS |
|
from open_webui.retrieval.web.main import SearchResult |
|
|
|
log = logging.getLogger(__name__) |
|
log.setLevel(SRC_LOG_LEVELS["RAG"]) |
|
|
|
EXA_API_BASE = "https://api.exa.ai" |
|
|
|
|
|
@dataclass |
|
class ExaResult: |
|
url: str |
|
title: str |
|
text: str |
|
|
|
|
|
def search_exa( |
|
api_key: str, |
|
query: str, |
|
count: int, |
|
filter_list: Optional[list[str]] = None, |
|
) -> list[SearchResult]: |
|
"""Search using Exa Search API and return the results as a list of SearchResult objects. |
|
|
|
Args: |
|
api_key (str): A Exa Search API key |
|
query (str): The query to search for |
|
count (int): Number of results to return |
|
filter_list (Optional[list[str]]): List of domains to filter results by |
|
""" |
|
log.info(f"Searching with Exa for query: {query}") |
|
|
|
headers = {"Authorization": f"Bearer {api_key}", "Content-Type": "application/json"} |
|
|
|
payload = { |
|
"query": query, |
|
"numResults": count or 5, |
|
"includeDomains": filter_list, |
|
"contents": {"text": True, "highlights": True}, |
|
"type": "auto", |
|
} |
|
|
|
try: |
|
response = requests.post( |
|
f"{EXA_API_BASE}/search", headers=headers, json=payload |
|
) |
|
response.raise_for_status() |
|
data = response.json() |
|
|
|
results = [] |
|
for result in data["results"]: |
|
results.append( |
|
ExaResult( |
|
url=result["url"], |
|
title=result["title"], |
|
text=result["text"], |
|
) |
|
) |
|
|
|
log.info(f"Found {len(results)} results") |
|
return [ |
|
SearchResult( |
|
link=result.url, |
|
title=result.title, |
|
snippet=result.text, |
|
) |
|
for result in results |
|
] |
|
except Exception as e: |
|
log.error(f"Error searching Exa: {e}") |
|
return [] |
|
|