Skip to main content

如何创建动态(自构建)链

前提条件

有时我们希望在运行时根据链的输入构建链的部分(路由是最常见的例子)。我们可以利用 RunnableLambda 的一个非常有用的特性来创建动态链,即如果 RunnableLambda 返回一个 Runnable,那么该 Runnable 会被直接调用。让我们看一个例子。

pip install -qU langchain-openai
import getpass
import os

os.environ["OPENAI_API_KEY"] = getpass.getpass()

from langchain_openai import ChatOpenAI

llm = ChatOpenAI(model="gpt-4o-mini")
# | echo: false

from langchain_anthropic import ChatAnthropic

llm = ChatAnthropic(model="claude-3-sonnet-20240229")
from operator import itemgetter

from langchain_core.output_parsers import StrOutputParser
from langchain_core.prompts import ChatPromptTemplate
from langchain_core.runnables import Runnable, RunnablePassthrough, chain

contextualize_instructions = """根据聊天记录将最新的用户问题转换为独立的问题。不要回答问题,只返回问题,不要有其他描述性文字。"""
contextualize_prompt = ChatPromptTemplate.from_messages(
[
("system", contextualize_instructions),
("placeholder", "{chat_history}"),
("human", "{question}"),
]
)
contextualize_question = contextualize_prompt | llm | StrOutputParser()

qa_instructions = (
"""根据以下上下文回答用户问题:\n\n{context}。"""
)
qa_prompt = ChatPromptTemplate.from_messages(
[("system", qa_instructions), ("human", "{question}")]
)


@chain
def contextualize_if_needed(input_: dict) -> Runnable:
if input_.get("chat_history"):
# 注意:这是返回另一个 Runnable,而不是实际输出。
return contextualize_question
else:
return RunnablePassthrough() | itemgetter("question")


@chain
def fake_retriever(input_: dict) -> str:
return "2024年埃及的总人口约为1.11亿"


full_chain = (
RunnablePassthrough.assign(question=contextualize_if_needed).assign(
context=fake_retriever
)
| qa_prompt
| llm
| StrOutputParser()
)

full_chain.invoke(
{
"question": "埃及怎么样",
"chat_history": [
("human", "印度尼西亚的人口是多少"),
("ai", "大约2.76亿"),
],
}
)
"根据提供的上下文,2024年埃及的总人口预计约为1.11亿。"

关键在于 contextualize_if_needed 返回另一个 Runnable,而不是实际输出。这个返回的 Runnable 在执行完整链时会被运行。

查看跟踪记录,我们可以看到,由于传入了 chat_history,我们在完整链中执行了 contextualize_question 链:https://smith.langchain.com/public/9e0ae34c-4082-4f3f-beed-34a2a2f4c991/r

请注意,返回的 Runnable 的流式处理、批处理等能力都得到了保留

for chunk in contextualize_if_needed.stream(
{
"question": "埃及怎么样",
"chat_history": [
("human", "印度尼西亚的人口是多少"),
("ai", "大约2.76亿"),
],
}
):
print(chunk)
埃及的
人口
是多少


此页面是否有帮助?


您还可以留下详细的反馈 在 GitHub 上