Spaces:
Running
LIBRAG for Boston Public Library
Problem
Our client, the Digital Repository Services Manager from the Boston Public Library (BPL), identified a need for a search function that enhances the accessibility and usability of the library's extensive digital resources. The goal is to implement a search function capable of understanding natural language queries and returning highly relevant documents from the BPL's vast database.
To meet this requirement, we are tasked with developing a demoable model that showcases this functionality. Our chosen approach leverages Retrieval-Augmented Generation (RAG) to provide an AI-driven, intuitive search experience.
Solution
We are developing a Retrieval-Augmented Generation (RAG) model tailored to the Boston Public Library's needs. This solution will:
- Enhance search performance: Provide accurate and relevant responses to natural language queries.
- Improve user experience: Deliver intuitive interactions that bridge the gap between traditional library systems and modern AI solutions.
- Showcase results through a user-friendly interface: Display the LLM response alongside:
- The documents used for generating the response.
- Direct URLs to the source documents.
Our prototype uses a simplified RAG pipeline built with LangChain, designed to retrieve from a subset of the library's dataset.
How the Database Is Built
The Digital Commonwealth database is vast and diverse, containing a wide variety of digital assets:
- Total Items: ~1.3 million items, including text, video, and audio.
- Full-Text Documents: A subset of the total items, ~147,000 OCR documents.
- Metadata: JSON files containing up to 135 description fields.
- File Types:
- Still Images: Most prevalent type, exceeding 600,000.
- Text Files: Second most common type.
- Other File Types: Fewer than 100,000 items, including notated music and manuscripts.
- Text Abstracts: Almost all images include textual descriptions, providing valuable metadata for retrieval.
Our solution ingested metadata items through the Digital Commonwealth API and embedded them for querying.
What is RAG?
Retrieval-Augmented Generation (RAG) combines information retrieval and generative AI to answer queries over a bespoke dataset. Here's how it works:
- Query Processing: Transform the user's natural language query into a format suitable for retrieval and embedding.
- Document Retrieval:
- Break down large documents into smaller chunks (chunking) to enhance granularity.
- Use embeddings to represent both the query and chunks as vectors.
- Retrieve relevant texts based on similarity scores (e.g., cosine similarity).
- Content Integration:
- Select and rank the most relevant chunks to answer the query.
- Response Generation: Use a large language model (LLM) to generate a response informed by the retrieved documents.
Key Components of RAG:
- Embedding Model: Converts text into numerical vectors that capture semantic meaning.
- Vector Store: A database to store and retrieve embeddings for similarity searches.
- Rerankig Algorithm: Determines priority of retrieved documents from the vectorstore (a second retrieval of sorts).
Where We Started
Our initial prototype used:
- LangChain for building the RAG pipeline.
- LangServe for serving the UI.
- Chroma as the vector store.
- OpenAI embeddings for text representation.
- GPT-4o as the large language model.
Challenges
Throughout development, we encountered several issues:
- Limited Customization: LangServe's UI was not sufficiently customizable.
- Lightweight Vector Store: Chroma and FAISS couldn't handle our large dataset effectively.
- Local Vector Store: A local vectorstore required the local storage of our large data and were cumbersome to load.
- High Costs: OpenAI embeddings were too expensive for a database of this size. Pinecone expenses grow quickly so we used a subset of our data.
Our Final Solution
We refined our approach to address these challenges:
- Embedding Model: Switched to
all-MiniLM-L6-v2
, a free, open-source model from Hugging Face, offering excellent performance. - LLM: Chose
4o-mini
, a cost-effective alternative with sufficient capabilities. - Vector Store: Adopted Pinecone vectorstore due to its accessibility via API given the size of the data.
- UI Development: Using Streamlit for UI due to its easy integration through python and compatibility with Huggingface Spaces.
Deployment
The final RAG model is hosted on Hugging Face, ensuring accessibility and reliability.
Our current solution externally hosts data on Pinecone and calls GPT-4o-mini.
Ethical Assessment
AI systems bring inherent risks that require careful management:
- Biased Document Retrieval:
- Risk: The model may retrieve biased documents depending on the query.
- Solution: Present to the user that this is from doocuments retrieved, not a response on behalf of the BPL.
- Conversational Hallucinations:
- Risk: The AI may generate responses not rooted in the database.
- Solution: Restrict the system to generate responses strictly based on retrieved documents.
License
This project is licensed under the MIT License, allowing others to freely integrate, modify, and build upon our work.
Replication
Let's get it running. For this use-case, we will only be ingesting the metadata associated with each record. Note that the general infrastructure would work on the OCR text as well.
Begin by creating a python virtual environment
Make sure you are at least in Python 3.12.4
python -m venv .venv
source .venv/bin/activate (Mac)
or
.venv/Scripts/Activate (Windows)
python -m pip install -r requirements.txt
Once you have your environment, it is time to begin running scripts!
Loading the BPL Digital Commonwealth metadata BEGIN = page number to start on END = page number to end on (non-inclusive) Pages each contain 100 items from the digital repository
python load_script.py <BEGIN> <END>
This will load the specified chunk of items from the Digital Commonwealth API, outputting to a file called "out<BEGIN>_<END>.json" in your current directory.
Next, we need to initialize our vectorstore
Initializing our Vectorstore
There are many vectorstore options to choose from. We ultimately decided on Pinecone due to the size of the data, which made local vectorstores inhibitively slow.
For our approach:
Make a Pinecone.io account
Navigate to Databases > Indexes and make a serverless Index. This is where we will store our embeddings. Make the length of each vector equal to whatever embeddings model you use. We used all-MiniLM-L6-v2, so our vector length was 384.
You will also want to create and store a Pinecone API Key in your .env file.
PINECONE_API_KEY = <your-api-key-here>
Embedding our data BEGIN = Start index of the metadata to begin embedding END = Final index of the metadata to stop embedding (non-inclusive) PATH = Path to your .json file of metadata INDEX = Name of your pinecone index
python load_pinecone.py <BEGIN> <END> <PATH> <INDEX>
Check to make sure the embeddings are being loaded into your index.
Running the App Now that you have your embeddings, you're ready for the main event!
Our implementation uses GPT-4o-mini from OpenAI, however you could fit in your own model to the RAG.py file. With our implementation, you will need to enter an openai api key into the .env file.
Also, make sure to replace the variable INDEX_NAME in streamlit-app.py with the name of your index.
OPENAI_API_KEY = "<you-api-key>"
Once you do that, you're ready to run.
streamlit run streamlit_app.py
This will run the app on port 8501. When querying, please be patient, sometimes the retrieval and re-ranking process is slow depending on how much data you embedded.