Skip to main content

人类中心论聊天

本笔记涵盖了如何开始使用人类中心论聊天模型。

设置

有关设置说明,请参阅人类中心平台页面的安装和环境设置部分。

%pip install -qU langchain-anthropic

环境设置

我们需要获取一个人类中心 API密钥,并设置 ANTHROPIC_API_KEY 环境变量:

import os
from getpass import getpass
os.environ["ANTHROPIC_API_KEY"] = getpass()

提供的代码假定您的 ANTHROPIC_API_KEY 已设置在环境变量中。如果您想手动指定您的 API密钥并选择不同的模型,您可以使用以下代码:

chat = ChatAnthropic(temperature=0, api_key="YOUR_API_KEY", model_name="claude-3-opus-20240229")

在这些演示中,我们将使用 Claude 3 Opus 模型,您还可以使用 Sonnet 模型的发布版本,其名称为 claude-3-sonnet-20240229

您可以在这里查看模型比较文档。

from langchain_anthropic import ChatAnthropic
from langchain_core.prompts import ChatPromptTemplate
chat = ChatAnthropic(temperature=0, model_name="claude-3-opus-20240229")
system = (
"您是一个有帮助的助手,可以将 {input_language} 翻译成 {output_language}。"
)
human = "{text}"
prompt = ChatPromptTemplate.from_messages([("system", system), ("human", human)])
chain = prompt | chat
chain.invoke(
{
"input_language": "英语",
"output_language": "韩语",
"text": "我喜欢Python",
}
)
AIMessage(content='저는 파이썬을 사랑합니다。\n\n翻译:\nI love Python.')

ChatAnthropic 还支持异步和流式功能:

chat = ChatAnthropic(temperature=0, model_name="claude-3-opus-20240229")
prompt = ChatPromptTemplate.from_messages([("human", "告诉我一个关于 {topic} 的笑话")])
chain = prompt | chat
await chain.ainvoke({"topic": "熊"})
AIMessage(content='当然,这是一个关于熊的笑话:\n\n一只熊走进酒吧对酒保说:“我要一品脱啤酒和一..........包花生。”\n\n酒保问:“为什么有这么长的停顿?”\n\n熊回答:“我不知道,我一直都有这个习惯!”')
chat = ChatAnthropic(temperature=0.3, model_name="claude-3-opus-20240229")
prompt = ChatPromptTemplate.from_messages(
[("human", "给我列举日本著名的旅游景点")]
)
chain = prompt | chat
for chunk in chain.stream({}):
print(chunk.content, end="", flush=True)
这是日本著名的旅游景点列表:
1. 东京晴空塔(东京)
2. 浅草寺(东京)
3. 明治神宫(东京)
4. 东京迪士尼海洋(浦安市,千叶县)
5. 伏见稻荷大社(京都)
6. 金阁寺(京都)
7. 清水寺(京都)
8. 二条城(京都)
9. 大阪城(大阪)
10. 道顿堀(大阪)
11. 广岛和平纪念公园(广岛)
12. 厳岛神社(宫岛,广岛)
13. 姬路城(姬路)
14. 大仏寺(奈良)
15. 奈良公园(奈良)
16. 富士山(静冈县和山梨县)
17.

[Beta] 工具调用

使用人类中心的工具调用或工具使用 API,您可以定义模型调用的工具。这对于构建使用工具链和代理非常有用,也可以从模型获得结构化输出。

note

人类中心的工具调用功能仍处于测试阶段。

bind_tools()

通过 ChatAnthropic.bind_tools,我们可以轻松地传递 Pydantic 类、字典模式、LangChain 工具,甚至函数作为模型的工具。在幕后,这些被转换为人类中心工具模式,看起来像这样:

{
"name": "...",
"description": "...",
"input_schema": {...} # JSONSchema
}

并在每次模型调用中传递。

from langchain_core.pydantic_v1 import BaseModel, Field
llm = ChatAnthropic(model="claude-3-opus-20240229", temperature=0)
class GetWeather(BaseModel):
"""获取给定位置的当前天气"""
location: str = Field(..., description="城市和州,例如旧金山,加利福尼亚州")
llm_with_tools = llm.bind_tools([GetWeather])
ai_msg = llm_with_tools.invoke(
"旧金山的天气如何",
)
ai_msg
AIMessage(content=[{'text': '<thinking>\n用户询问特定位置(旧金山)的当前天气情况,回答这个问题的相关工具是 GetWeather 函数。\n\n查看 GetWeather 的参数:\n- location(必填):用户在查询中直接提供了位置 - "San Francisco"\n\n由于存在必填参数 "location",我们可以继续调用 GetWeather 函数。\n</thinking>', 'type': 'text'}, {'id': 'toolu_01StzxdWQSZhAMbR1CCchQV9', 'input': {'location': 'San Francisco, CA'}, 'name': 'GetWeather', 'type': 'tool_use'}], response_metadata={'id': 'msg_01HepCTzqXJed5iNuLgV1VCZ', 'model': 'claude-3-opus-20240229', 'stop_reason': 'tool_use', 'stop_sequence': None, 'usage': {'input_tokens': 487, 'output_tokens': 143}}, id='run-1a1b3289-ba2c-47ae-8be1-8929d7cc547e-0', tool_calls=[{'name': 'GetWeather', 'args': {'location': 'San Francisco, CA'}, 'id': 'toolu_01StzxdWQSZhAMbR1CCchQV9'}])
ai_msg.content
[{'text': '<thinking>\n用户询问特定位置(旧金山)的当前天气情况。回答这个问题的相关工具是 GetWeather 函数。\n\n查看 GetWeather 的参数:\n- location(必填):用户在查询中直接提供了位置 - “旧金山”\n\n由于提供了必需的“location”参数,我们可以继续调用 GetWeather 函数。\n</thinking>',
'type': 'text'},
{'id': 'toolu_01StzxdWQSZhAMbR1CCchQV9',
'input': {'location': 'San Francisco, CA'},
'name': 'GetWeather',
'type': 'tool_use'}]
ai_msg.tool_calls
[{'name': 'GetWeather',
'args': {'location': 'San Francisco, CA'},
'id': 'toolu_01StzxdWQSZhAMbR1CCchQV9'}]
tip

ChatAnthropic 模型输出始终是一个单一的 AI 消息,可以是单个字符串或内容块列表。内容块可以是文本块或工具使用块。每种类型都可以有多个,它们可以交替出现。

强制调用工具

默认情况下,模型可以选择是否调用任何工具。要强制模型调用至少一个工具,我们可以指定 bind_tools(..., tool_choice="any"),要强制模型调用特定工具,可以传入该工具名称 bind_tools(..., tool_choice="GetWeather")

llm_with_force_tools = llm.bind_tools([GetWeather], tool_choice="GetWeather")
# 请注意,即使消息与工具无关,模型仍将返回工具调用。
llm_with_force_tools.invoke("this doesn't really require tool use").tool_calls
[{'name': 'GetWeather',
'args': {'location': '<UNKNOWN>'},
'id': 'toolu_01DwWjKzHPs6EHCUPxsGm9bN'}]

解析工具调用

如果需要,langchain_anthropic.output_parsers.ToolsOutputParser 可以轻松将 Anthropic AI 消息中的工具调用解析为 Pydantic 对象:

from langchain_anthropic.output_parsers import ToolsOutputParser
parser = ToolsOutputParser(pydantic_schemas=[GetWeather])
chain = llm_with_tools | parser
chain.invoke("What is the weather like in nyc, la, sf and cleveland")
[GetWeather(location='New York City, NY'),
GetWeather(location='Los Angeles, CA'),
GetWeather(location='San Francisco, CA'),
GetWeather(location='Cleveland, OH')]

with_structured_output()

BaseChatModel.with_structured_output 接口 使得从聊天模型获取结构化输出变得轻而易举。您可以使用 ChatAnthropic.with_structured_output(在底层使用工具调用)来让模型更可靠地返回特定格式的输出:

structured_llm = llm.with_structured_output(GetWeather)
structured_llm.invoke(
"what is the weather like in San Francisco",
)
GetWeather(location='San Francisco, CA')

使用

llm.with_structured_output(GetWeather)

llm.bind_tools([GetWeather]) | ToolsOutputParser(pydantic_schemas=[GetWeather])

的主要区别在于,前者将仅返回第一个 GetWeather 调用,而后者将返回一个列表。

将工具结果传递给模型

我们可以使用带有适当的 tool_call_idToolMessage 将工具结果传递回模型:

from langchain_core.messages import AIMessage, HumanMessage, ToolMessage
messages = [
HumanMessage("What is the weather like in San Francisco"),
AIMessage(
content=[
{
"text": '<thinking>\n基于用户的问题,调用相关函数 GetWeather 需要“location”参数。\n\n用户直接指定了位置为“旧金山”。由于旧金山是一个众所周知的城市,我可以合理推断他们指的是加利福尼亚州的旧金山,而不需要指定州。\n\n所有必需的参数都已提供,因此我可以进行 API 调用。\n</thinking>',
"type": "text",
},
{
"type": "tool_use",
"id": "toolu_01SCgExKzQ7eqSkMHfygvYuu",
"name": "GetWeather",
"input": {"location": "San Francisco, CA"},
"text": None,
},
],
),
ToolMessage(
"Rain. High 54F. Winds SW at 15 to 25 mph. Chance of rain 100%.",
tool_call_id="toolu_01SCgExKzQ7eqSkMHfygvYuu",
),
]
llm_with_tools.invoke(messages)
AIMessage(content='根据调用 GetWeather 函数,旧金山,加利福尼亚的天气是:\n雨,最高气温 54°F,来自西南方向的风速为 15-25 mph。有 100% 的降雨几率。', response_metadata={'id': 'msg_01J7nWVRPPTgae4eDpf9yR3M', 'model': 'claude-3-opus-20240229', 'stop_reason': 'end_turn', 'stop_sequence': None, 'usage': {'input_tokens': 670, 'output_tokens': 56}}, id='run-44fcd34f-9c24-464f-94dd-63bd0d22870d-0')

流式处理

::: {.callout-warning}

Anthropic 目前不支持流式处理工具调用。尝试进行流式处理将会得到一条最终消息。

:::

list(llm_with_tools.stream("旧金山的天气如何"))
/Users/bagatur/langchain/libs/partners/anthropic/langchain_anthropic/chat_models.py:328: UserWarning: stream: Tool use is not yet supported in streaming mode.
warnings.warn("stream: Tool use is not yet supported in streaming mode.")
[AIMessage(content=[{'text': '<thinking>\n用户正在询问特定位置(旧金山)的当前天气情况。GetWeather 函数是回答这个请求的相关工具,因为它可以返回给定位置的当前天气情况。\n\nGetWeather 函数有一个必需的参数:\nlocation: 城市和州,例如旧金山,加利福尼亚\n\n用户在请求中提供了城市旧金山。他们没有指定州,但可以合理推断他们指的是加利福尼亚州的旧金山,因为那是这个名字最为人熟知的城市。\n\n由于用户提供了必需的位置参数,我们可以继续调用 GetWeather 函数。\n</thinking>', 'type': 'text'}, {'text': None, 'type': 'tool_use', 'id': 'toolu_01V9ZripoQzuY8HubspJy6fP', 'name': 'GetWeather', 'input': {'location': 'San Francisco, CA'}}], id='run-b825206b-5b6b-48bc-ad8d-802dee310c7f')]

多模态

Anthropic 的 Claude-3 模型兼容图像和文本输入。您可以按以下方式使用:

# 以 base64 字符串形式打开 ../../../static/img/brand/wordmark.png
import base64
from pathlib import Path
from IPython.display import HTML
img_path = Path("../../../static/img/brand/wordmark.png")
img_base64 = base64.b64encode(img_path.read_bytes()).decode("utf-8")
# 在笔记本中显示 base64 图像
HTML(f'<img src="data:image/png;base64,{img_base64}"/>')
<img src=""/> 
from langchain_core.messages import HumanMessage
chat = ChatAnthropic(model="claude-3-opus-20240229")
messages = [
HumanMessage(
content=[
{
"type": "image_url",
"image_url": {
# langchain logo
"url": f"data:image/png;base64,{img_base64}", # noqa: E501
},
},
{"type": "text", "text": "这个标志是做什么用的?"},
]
)
]
chat.invoke(messages)
AIMessage(content='这个标志是为 LangChain 设计的,根据标志的名字和极简主义设计风格,它似乎是一种基于某种软件或技术平台,标志上有一只鸟的轮廓(可能是鹰或隼),并且公司名称使用了简单现代的字体。')

Was this page helpful?


You can leave detailed feedback on GitHub.