Retrievers
Head to Integrations for documentation on built-in integrations with retrieval providers.
A retriever is an interface that returns documents given an unstructured query. It is more general than a vector store. A retriever does not need to be able to store documents, only to return (or retrieve) it. Vector stores can be used as the backbone of a retriever, but there are other types of retrievers as well.
Get started
The public API of the BaseRetriever
class in LangChain.js is as follows:
export abstract class BaseRetriever {
abstract getRelevantDocuments(query: string): Promise<Document[]>;
}
It's that simple! You can call getRelevantDocuments
to retrieve documents relevant to a query, where "relevance" is defined by
the specific retriever object you are calling.
Of course, we also help construct what we think useful Retrievers are. The main type of Retriever in LangChain is a vector store retriever. We will focus on that here.
Note: Before reading, it's important to understand what a vector store is.
This example showcases question answering over documents. We have chosen this as the example for getting started because it nicely combines a lot of different elements (Text splitters, embeddings, vectorstores) and then also shows how to use them in a chain.
Question answering over documents consists of four steps:
- Create an index
- Create a Retriever from that index
- Create a question answering chain
- Ask questions!
Each of the steps has multiple sub steps and potential configurations, but we'll go through one common flow using HNSWLib, a local vector store. This assumes you're using Node, but you can swap in another integration if necessary.
First, install the required dependency:
- npm
- Yarn
- pnpm
npm install -S hnswlib-node
yarn add hnswlib-node
pnpm add hnswlib-node
You can download the state_of_the_union.txt
file here.
import { HNSWLib } from "langchain/vectorstores/hnswlib";
import { OpenAIEmbeddings } from "langchain/embeddings/openai";
import { RecursiveCharacterTextSplitter } from "langchain/text_splitter";
import * as fs from "fs";
import {
RunnablePassthrough,
RunnableSequence,
} from "langchain/schema/runnable";
import { StringOutputParser } from "langchain/schema/output_parser";
import {
ChatPromptTemplate,
HumanMessagePromptTemplate,
SystemMessagePromptTemplate,
} from "langchain/prompts";
import { ChatOpenAI } from "langchain/chat_models/openai";
import { formatDocumentsAsString } from "langchain/util/document";
// Initialize the LLM to use to answer the question.
const model = new ChatOpenAI({});
const text = fs.readFileSync("state_of_the_union.txt", "utf8");
const textSplitter = new RecursiveCharacterTextSplitter({ chunkSize: 1000 });
const docs = await textSplitter.createDocuments([text]);
// Create a vector store from the documents.
const vectorStore = await HNSWLib.fromDocuments(docs, new OpenAIEmbeddings());
// Initialize a retriever wrapper around the vector store
const vectorStoreRetriever = vectorStore.asRetriever();
// Create a system & human prompt for the chat model
const SYSTEM_TEMPLATE = `Use the following pieces of context to answer the question at the end.
If you don't know the answer, just say that you don't know, don't try to make up an answer.
----------------
{context}`;
const messages = [
SystemMessagePromptTemplate.fromTemplate(SYSTEM_TEMPLATE),
HumanMessagePromptTemplate.fromTemplate("{question}"),
];
const prompt = ChatPromptTemplate.fromMessages(messages);
const chain = RunnableSequence.from([
{
context: vectorStoreRetriever.pipe(formatDocumentsAsString),
question: new RunnablePassthrough(),
},
prompt,
model,
new StringOutputParser(),
]);
const answer = await chain.invoke(
"What did the president say about Justice Breyer?"
);
console.log({ answer });
/*
{
answer: 'The president thanked Justice Stephen Breyer for his service and honored him for his dedication to the country.'
}
*/
API Reference:
- HNSWLib from
langchain/vectorstores/hnswlib
- OpenAIEmbeddings from
langchain/embeddings/openai
- RecursiveCharacterTextSplitter from
langchain/text_splitter
- RunnablePassthrough from
langchain/schema/runnable
- RunnableSequence from
langchain/schema/runnable
- StringOutputParser from
langchain/schema/output_parser
- ChatPromptTemplate from
langchain/prompts
- HumanMessagePromptTemplate from
langchain/prompts
- SystemMessagePromptTemplate from
langchain/prompts
- ChatOpenAI from
langchain/chat_models/openai
- formatDocumentsAsString from
langchain/util/document
Let's walk through what's happening here.
We first load a long text and split it into smaller documents using a text splitter. We then load those documents (which also embeds the documents using the passed
OpenAIEmbeddings
instance) into HNSWLib, our vector store, creating our index.Though we can query the vector store directly, we convert the vector store into a retriever to return retrieved documents in the right format for the question answering chain.
We initialize a
RetrievalQAChain
with the.fromLLM
method, which we'll call later in step 4.We ask questions!
See the individual sections for deeper dives on specific retrievers.