2024-03-18 00:44:45 +08:00
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
|
|
import typing
|
2024-03-18 11:22:26 +08:00
|
|
|
|
import traceback
|
2024-03-18 00:44:45 +08:00
|
|
|
|
|
|
|
|
|
import anthropic
|
|
|
|
|
|
|
|
|
|
from .. import api, entities, errors
|
|
|
|
|
|
|
|
|
|
from .. import api, entities, errors
|
|
|
|
|
from ....core import entities as core_entities
|
|
|
|
|
from ... import entities as llm_entities
|
|
|
|
|
from ...tools import entities as tools_entities
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@api.requester_class("anthropic-messages")
|
|
|
|
|
class AnthropicMessages(api.LLMAPIRequester):
|
|
|
|
|
"""Anthropic Messages API 请求器"""
|
|
|
|
|
|
|
|
|
|
client: anthropic.AsyncAnthropic
|
|
|
|
|
|
|
|
|
|
async def initialize(self):
|
|
|
|
|
self.client = anthropic.AsyncAnthropic(
|
|
|
|
|
api_key="",
|
|
|
|
|
base_url=self.ap.provider_cfg.data['requester']['anthropic-messages']['base-url'],
|
|
|
|
|
timeout=self.ap.provider_cfg.data['requester']['anthropic-messages']['timeout'],
|
|
|
|
|
proxies=self.ap.proxy_mgr.get_forward_proxies()
|
|
|
|
|
)
|
|
|
|
|
|
2024-05-14 22:20:31 +08:00
|
|
|
|
async def call(
|
2024-03-18 00:44:45 +08:00
|
|
|
|
self,
|
2024-05-14 22:20:31 +08:00
|
|
|
|
model: entities.LLMModelInfo,
|
|
|
|
|
messages: typing.List[llm_entities.Message],
|
|
|
|
|
funcs: typing.List[tools_entities.LLMFunction] = None,
|
|
|
|
|
) -> llm_entities.Message:
|
|
|
|
|
self.client.api_key = model.token_mgr.get_token()
|
2024-03-18 00:44:45 +08:00
|
|
|
|
|
|
|
|
|
args = self.ap.provider_cfg.data['requester']['anthropic-messages']['args'].copy()
|
2024-05-14 22:20:31 +08:00
|
|
|
|
args["model"] = model.name if model.model_name is None else model.model_name
|
2024-03-18 00:44:45 +08:00
|
|
|
|
|
2024-05-14 22:20:31 +08:00
|
|
|
|
req_messages = [
|
|
|
|
|
m.dict(exclude_none=True) for m in messages if m.content.strip() != ""
|
|
|
|
|
]
|
2024-03-18 00:44:45 +08:00
|
|
|
|
|
2024-05-14 22:20:31 +08:00
|
|
|
|
# 删除所有 role=system & content='' 的消息
|
2024-03-18 00:44:45 +08:00
|
|
|
|
req_messages = [
|
|
|
|
|
m for m in req_messages if not (m["role"] == "system" and m["content"].strip() == "")
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
# 检查是否有 role=system 的消息,若有,改为 role=user,并在后面加一个 role=assistant 的消息
|
|
|
|
|
system_role_index = []
|
|
|
|
|
for i, m in enumerate(req_messages):
|
|
|
|
|
if m["role"] == "system":
|
|
|
|
|
system_role_index.append(i)
|
|
|
|
|
m["role"] = "user"
|
|
|
|
|
|
|
|
|
|
if system_role_index:
|
|
|
|
|
for i in system_role_index[::-1]:
|
|
|
|
|
req_messages.insert(i + 1, {"role": "assistant", "content": "Okay, I'll follow."})
|
|
|
|
|
|
2024-03-18 11:06:40 +08:00
|
|
|
|
# 忽略掉空消息,用户可能发送空消息,而上层未过滤
|
|
|
|
|
req_messages = [
|
|
|
|
|
m for m in req_messages if m["content"].strip() != ""
|
|
|
|
|
]
|
|
|
|
|
|
2024-03-18 00:44:45 +08:00
|
|
|
|
args["messages"] = req_messages
|
|
|
|
|
|
2024-03-18 11:22:26 +08:00
|
|
|
|
try:
|
|
|
|
|
resp = await self.client.messages.create(**args)
|
|
|
|
|
|
2024-05-14 22:20:31 +08:00
|
|
|
|
return llm_entities.Message(
|
2024-03-18 11:22:26 +08:00
|
|
|
|
content=resp.content[0].text,
|
|
|
|
|
role=resp.role
|
|
|
|
|
)
|
|
|
|
|
except anthropic.AuthenticationError as e:
|
|
|
|
|
raise errors.RequesterError(f'api-key 无效: {e.message}')
|
|
|
|
|
except anthropic.BadRequestError as e:
|
|
|
|
|
raise errors.RequesterError(str(e.message))
|
|
|
|
|
except anthropic.NotFoundError as e:
|
|
|
|
|
if 'model: ' in str(e):
|
|
|
|
|
raise errors.RequesterError(f'模型无效: {e.message}')
|
|
|
|
|
else:
|
2024-05-14 22:20:31 +08:00
|
|
|
|
raise errors.RequesterError(f'请求地址无效: {e.message}')
|