Files
LangBot/pkg/provider/modelmgr/apis/anthropicmsgs.py

114 lines
3.8 KiB
Python
Raw Normal View History

2024-03-17 12:44:45 -04:00
from __future__ import annotations
import typing
2024-03-17 23:22:26 -04:00
import traceback
2024-03-17 12:44:45 -04: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
2024-05-22 20:09:29 +08:00
from ....utils import image
2024-03-17 12:44:45 -04:00
@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()
)
async def call(
2024-03-17 12:44:45 -04:00
self,
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-17 12:44:45 -04:00
args = self.ap.provider_cfg.data['requester']['anthropic-messages']['args'].copy()
args["model"] = model.name if model.model_name is None else model.model_name
2024-03-17 12:44:45 -04:00
2024-05-15 21:40:18 +08:00
# 处理消息
# system
system_role_message = None
for i, m in enumerate(messages):
if m.role == "system":
system_role_message = m
messages.pop(i)
break
if isinstance(system_role_message, llm_entities.Message) \
and isinstance(system_role_message.content, str):
args['system'] = system_role_message.content
req_messages = []
for m in messages:
if isinstance(m.content, str) and m.content.strip() != "":
req_messages.append(m.dict(exclude_none=True))
elif isinstance(m.content, list):
2024-05-22 20:09:29 +08:00
# m.content = [
# c for c in m.content if c.type == "text"
# ]
2024-05-15 21:40:18 +08:00
2024-05-22 20:09:29 +08:00
# if len(m.content) > 0:
# req_messages.append(m.dict(exclude_none=True))
msg_dict = m.dict(exclude_none=True)
for i, ce in enumerate(m.content):
if ce.type == "image_url":
base64_image, image_format = await image.qq_image_url_to_base64(ce.image_url.url)
2024-05-22 20:09:29 +08:00
alter_image_ele = {
"type": "image",
"source": {
"type": "base64",
"media_type": f"image/{image_format}",
"data": base64_image
2024-05-22 20:09:29 +08:00
}
}
msg_dict["content"][i] = alter_image_ele
req_messages.append(msg_dict)
2024-03-17 23:06:40 -04:00
2024-03-17 12:44:45 -04:00
args["messages"] = req_messages
2024-05-22 20:09:29 +08:00
# anthropic的tools处在beta阶段sdk不稳定故暂时不支持
#
# if funcs:
# tools = await self.ap.tool_mgr.generate_tools_for_openai(funcs)
# if tools:
# args["tools"] = tools
2024-03-17 23:22:26 -04:00
try:
resp = await self.client.messages.create(**args)
return llm_entities.Message(
2024-03-17 23:22:26 -04: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:
raise errors.RequesterError(f'请求地址无效: {e.message}')