51 lines
2 KiB
JavaScript
51 lines
2 KiB
JavaScript
|
import { RunnableSequence, RunnableBranch, } from "@langchain/core/runnables";
|
||
|
import { StringOutputParser } from "@langchain/core/output_parsers";
|
||
|
/**
|
||
|
* Create a chain that takes conversation history and returns documents.
|
||
|
* If there is no `chat_history`, then the `input` is just passed directly to the
|
||
|
* retriever. If there is `chat_history`, then the prompt and LLM will be used
|
||
|
* to generate a search query. That search query is then passed to the retriever.
|
||
|
* @param {CreateHistoryAwareRetriever} params
|
||
|
* @returns An LCEL Runnable. The runnable input must take in `input`, and if there
|
||
|
* is chat history should take it in the form of `chat_history`.
|
||
|
* The Runnable output is a list of Documents
|
||
|
* @example
|
||
|
* ```typescript
|
||
|
* // yarn add langchain @langchain/openai
|
||
|
*
|
||
|
* import { ChatOpenAI } from "@langchain/openai";
|
||
|
* import { pull } from "langchain/hub";
|
||
|
* import { createHistoryAwareRetriever } from "langchain/chains/history_aware_retriever";
|
||
|
*
|
||
|
* const rephrasePrompt = await pull("langchain-ai/chat-langchain-rephrase");
|
||
|
* const llm = new ChatOpenAI({});
|
||
|
* const retriever = ...
|
||
|
* const chain = await createHistoryAwareRetriever({
|
||
|
* llm,
|
||
|
* retriever,
|
||
|
* rephrasePrompt,
|
||
|
* });
|
||
|
* const result = await chain.invoke({"input": "...", "chat_history": [] })
|
||
|
* ```
|
||
|
*/
|
||
|
export async function createHistoryAwareRetriever({ llm, retriever, rephrasePrompt, }) {
|
||
|
if (!rephrasePrompt.inputVariables.includes("input")) {
|
||
|
throw new Error(`Expected "input" to be a prompt variable, but got ${JSON.stringify(rephrasePrompt.inputVariables)}`);
|
||
|
}
|
||
|
const retrieveDocuments = RunnableBranch.from([
|
||
|
[
|
||
|
(input) => !input.chat_history || input.chat_history.length === 0,
|
||
|
RunnableSequence.from([(input) => input.input, retriever]),
|
||
|
],
|
||
|
RunnableSequence.from([
|
||
|
rephrasePrompt,
|
||
|
llm,
|
||
|
new StringOutputParser(),
|
||
|
retriever,
|
||
|
]),
|
||
|
]).withConfig({
|
||
|
runName: "history_aware_retriever",
|
||
|
});
|
||
|
return retrieveDocuments;
|
||
|
}
|