Search with Sources
This recipe shows how to use LLMs — in this case, GPT 4o mini — to answer questions using the web. Since LLMs often time hallucinate answers, it is important to fact check and verify the accuracy of the answer.
Users of Large Language Models (LLMs) often struggle to distinguish between factual content and potential hallucinations, leading to time-consuming fact-checking. By implementing source citation requirements, LLMs need to rely on verified information, thereby enhancing the accuracy of its responses and reducing the need for manual verification.
Setup
To set up our environment, first let's install all of the packages we will use:
!pip install "mirascope[openai]" beautifulsoup4import os
os.environ["OPENAI_API_KEY"] = "YOUR_API_KEY"
# Set the appropriate API key for the provider you're usingWe will need an API key for search:
- Nimble API Key or alternatively directly from Google Custom Search API.
Creating Google Search tool
We use Nimble since they provide an easy-to-use API for searching, but an alternative you can use is Google's Custom Search API. We first want to grab all the urls that are relevant to answer our question and then we take the contents of those urls, like so:
import requests
from bs4 import BeautifulSoup
NIMBLE_TOKEN = "YOUR_NIMBLE_API_KEY"
def nimble_google_search(query: str):
    """
    Use Nimble to get information about the query using Google Search.
    """
    url = "https://api.webit.live/api/v1/realtime/serp"
    headers = {
        "Authorization": f"Basic {NIMBLE_TOKEN}",
        "Content-Type": "application/json",
    }
    search_data = {
        "parse": True,
        "query": query,
        "search_engine": "google_search",
        "format": "json",
        "render": True,
        "country": "US",
        "locale": "en",
    }
    response = requests.get(url, json=search_data, headers=headers)
    data = response.json()
    results = data["parsing"]["entities"]["OrganicResult"]
    urls = [result.get("url", "") for result in results]
    search_results = {}
    for url in urls:
        content = get_content(url)
        search_results[url] = content
    return search_results
def get_content(url: str):
    data = []
    response = requests.get(url)
    content = response.content
    soup = BeautifulSoup(content, "html.parser")
    paragraphs = soup.find_all("p")
    for paragraph in paragraphs:
        data.append(paragraph.text)
    return "\n".join(data)Now that we have created our tool, it’s time to create our LLM call.
Creating the first call
For this call, we force the LLM to always use its tool which we will later chain.
from mirascope.core import openai, prompt_template
@openai.call(
    model="gpt-4o-mini",
    tools=[nimble_google_search],
    call_params={"tool_choice": "required"},
)
@prompt_template(
    """
    SYSTEM:
    You are a an expert at finding information on the web.
    Use the `nimble_google_search` function to find information on the web.
    Rewrite the question as needed to better find information on the web.
    USER:
    {question}
    """
)
def search(question: str): ...We ask the LLM to rewrite the question to make it more suitable for search.
Now that we have the necessary data to answer the user query and their sources, it’s time to extract all that information into a structured format using response_model
Extracting Search Results with Sources
As mentioned earlier, it is important to fact check all answers in case of hallucination, and the first step is to ask the LLM to cite its sources:
from pydantic import BaseModel, Field
class SearchResponse(BaseModel):
    sources: list[str] = Field(description="The sources of the results")
    answer: str = Field(description="The answer to the question")
@openai.call(model="gpt-4o-mini", response_model=list[SearchResponse])
@prompt_template(
    """
    SYSTEM:
    Extract the question, results, and sources to answer the question based on the results.
    Results:
    {results}
    USER:
    {question}
    """
)
def extract(question: str, results: str): ...and finally we create our run function to execute our chain:
def run(question: str):
    response = search(question)
    if tool := response.tool:
        output = tool.call()
        result = extract(question, output)
        return result
print(run("What is the average price of a house in the United States?"))- Journalism Assistant: Have the LLM do some research to quickly pull verifiable sources for blog posts and news articles.
- Education: Find and cite articles to help write academic papers.
- Technical Documentation: Generate code snippets and docs referencing official documentation.
When adapting this recipe, consider:
- Adding Tenacity retryfor more a consistent extraction.
- Use an LLM with web search tool to evaluate whether the answer produced is in the source.
- Experiment with different model providers and version for quality and accuracy of results.