从 MapRerankDocumentsChain 迁移
MapRerankDocumentsChain 实现了一种分析长文本的策略。策略如下:
- 将文本拆分为较小的文档;
- 将流程映射到文档集,其中流程包括生成分数;
- 按分数对结果进行排名并返回最大值。
此方案中的一个常见过程是使用文档中的上下文片段进行问答。强制模型生成分数及其答案有助于选择仅由相关上下文生成的答案。
LangGraph 实现允许为此问题合并工具调用和其他功能。下面我们将介绍两者MapRerankDocumentsChain以及一个简单的示例上相应的 LangGraph 实现,用于说明目的。
例
让我们看一个分析一组文档的示例。让我们使用以下 3 个文档:
from langchain_core.documents import Document
documents = [
    Document(page_content="Alice has blue eyes", metadata={"title": "book_chapter_2"}),
    Document(page_content="Bob has brown eyes", metadata={"title": "book_chapter_1"}),
    Document(
        page_content="Charlie has green eyes", metadata={"title": "book_chapter_3"}
    ),
]
API 参考:文档
遗产
详
下面我们展示了一个MapRerankDocumentsChain.为此,我们为问答任务定义了 prompt 模板,并实例化了一个 LLMChain 对象。我们定义如何将文档格式化到 Prompt 中,并确保各种 Prompt 中键之间的一致性。
from langchain.chains import LLMChain, MapRerankDocumentsChain
from langchain.output_parsers.regex import RegexParser
from langchain_core.prompts import PromptTemplate
from langchain_openai import OpenAI
document_variable_name = "context"
llm = OpenAI()
# The prompt here should take as an input variable the
# `document_variable_name`
# The actual prompt will need to be a lot more complex, this is just
# an example.
prompt_template = (
    "What color are Bob's eyes? "
    "Output both your answer and a score (1-10) of how confident "
    "you are in the format: <Answer>\nScore: <Score>.\n\n"
    "Provide no other commentary.\n\n"
    "Context: {context}"
)
output_parser = RegexParser(
    regex=r"(.*?)\nScore: (.*)",
    output_keys=["answer", "score"],
)
prompt = PromptTemplate(
    template=prompt_template,
    input_variables=["context"],
    output_parser=output_parser,
)
llm_chain = LLMChain(llm=llm, prompt=prompt)
chain = MapRerankDocumentsChain(
    llm_chain=llm_chain,
    document_variable_name=document_variable_name,
    rank_key="score",
    answer_key="answer",
)
response = chain.invoke(documents)
response["output_text"]
/langchain/libs/langchain/langchain/chains/llm.py:369: UserWarning: The apply_and_parse method is deprecated, instead pass an output parser directly to LLMChain.
  warnings.warn(
'Brown'
检查上述运行的 LangSmith 跟踪,我们可以看到 3 个 LLM 调用(每个文档 1 个),并且评分机制减轻了幻觉。
LangGraph
详
下面我们展示了这个过程的 LangGraph 实现。请注意,我们的模板是简化的,因为我们通过 .with_structured_output 方法将格式设置指令委托给聊天模型的工具调用功能。
在这里,我们遵循基本的 map-reduce 工作流来并行执行 LLM 调用。
我们需要安装langgraph:
pip install -qU langgraph
import operator
from typing import Annotated, List, TypedDict
from langchain_core.prompts import ChatPromptTemplate
from langchain_openai import ChatOpenAI
from langgraph.constants import Send
from langgraph.graph import END, START, StateGraph
class AnswerWithScore(TypedDict):
    answer: str
    score: Annotated[int, ..., "Score from 1-10."]
llm = ChatOpenAI(model="gpt-4o-mini", temperature=0)
prompt_template = "What color are Bob's eyes?\n\n" "Context: {context}"
prompt = ChatPromptTemplate.from_template(prompt_template)
# The below chain formats context from a document into a prompt, then
# generates a response structured according to the AnswerWithScore schema.
map_chain = prompt | llm.with_structured_output(AnswerWithScore)
# Below we define the components that will make up the graph
# This will be the overall state of the graph.
# It will contain the input document contents, corresponding
# answers with scores, and a final answer.
class State(TypedDict):
    contents: List[str]
    answers_with_scores: Annotated[list, operator.add]
    answer: str
# This will be the state of the node that we will "map" all
# documents to in order to generate answers with scores
class MapState(TypedDict):
    content: str
# Here we define the logic to map out over the documents
# We will use this an edge in the graph
def map_analyses(state: State):
    # We will return a list of `Send` objects
    # Each `Send` object consists of the name of a node in the graph
    # as well as the state to send to that node
    return [
        Send("generate_analysis", {"content": content}) for content in state["contents"]
    ]
# Here we generate an answer with score, given a document
async def generate_analysis(state: MapState):
    response = await map_chain.ainvoke(state["content"])
    return {"answers_with_scores": [response]}
# Here we will select the top answer
def pick_top_ranked(state: State):
    ranked_answers = sorted(
        state["answers_with_scores"], key=lambda x: -int(x["score"])
    )
    return {"answer": ranked_answers[0]}
# Construct the graph: here we put everything together to construct our graph
graph = StateGraph(State)
graph.add_node("generate_analysis", generate_analysis)
graph.add_node("pick_top_ranked", pick_top_ranked)
graph.add_conditional_edges(START, map_analyses, ["generate_analysis"])
graph.add_edge("generate_analysis", "pick_top_ranked")
graph.add_edge("pick_top_ranked", END)
app = graph.compile()
from IPython.display import Image
Image(app.get_graph().draw_mermaid_png())
result = await app.ainvoke({"contents": [doc.page_content for doc in documents]})
result["answer"]
{'answer': 'Bob has brown eyes.', 'score': 10}
检查上述运行的 LangSmith 跟踪,我们可以像以前一样看到三个 LLM 调用。使用模型的工具调用功能还使我们能够删除解析步骤。
后续步骤
有关使用 RAG 进行问答任务的更多信息,请参阅这些操作指南。
查看 LangGraph 文档,了解有关使用 LangGraph 构建的详细信息,包括有关 LangGraph 中 map-reduce 的详细信息的指南。