eigent/backend/camel/models/_utils.py
2026-03-31 17:20:08 +08:00

102 lines
3.6 KiB
Python

# ========= Copyright 2023-2026 @ CAMEL-AI.org. All Rights Reserved. =========
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
# ========= Copyright 2023-2026 @ CAMEL-AI.org. All Rights Reserved. =========
import re
import textwrap
from typing import Type
from pydantic import BaseModel
from camel.messages import OpenAIMessage
def extract_thinking_from_content(
content: str,
reasoning_content: str | None = None,
) -> tuple[str, str | None]:
r"""Extract ``<think>`` tags from content into reasoning_content.
Used for both pre-processing (stripping ``<think>`` tags from
input messages before sending to the model) and post-processing
(extracting ``<think>`` tags from model output into the
``reasoning_content`` field).
If the content contains ``<think>...</think>`` tags and
``reasoning_content`` is not already set, extracts the thinking
content and returns cleaned content with the reasoning.
Args:
content (str): The response content that may contain
``<think>`` tags.
reasoning_content (str | None): Existing reasoning content.
If already set, no extraction is performed.
(default: :obj:`None`)
Returns:
tuple[str, str | None]: A tuple of (cleaned_content,
reasoning_content). If no extraction was needed, returns
the original values unchanged.
"""
if reasoning_content is not None:
return content, reasoning_content
if '<think>' not in content or '</think>' not in content:
return content, reasoning_content
think_match = re.search(r'<think>(.*?)</think>', content, flags=re.DOTALL)
if think_match:
extracted = think_match.group(1).strip()
# Only set reasoning_content if there's actual content
if extracted:
reasoning_content = extracted
content = re.sub(
r'<think>.*?</think>', '', content, flags=re.DOTALL
).strip()
return content, reasoning_content
def try_modify_message_with_format(
message: OpenAIMessage,
response_format: Type[BaseModel] | None,
) -> None:
r"""Modifies the content of the message to include the instruction of using
the response format.
The message will not be modified in the following cases:
- response_format is None
- message content is not a string
- message role is assistant
Args:
response_format (Type[BaseModel] | None): The Pydantic model class.
message (OpenAIMessage): The message to be modified.
"""
if response_format is None:
return
if not isinstance(message["content"], str):
return
if message["role"] == "assistant":
return
json_schema = response_format.model_json_schema()
updated_prompt = textwrap.dedent(
f"""\
{message["content"]}
Please generate a JSON response adhering to the following JSON schema:
{json_schema}
Make sure the JSON response is valid and matches the EXACT structure defined in the schema. Your result should ONLY be a valid json object, WITHOUT ANY OTHER TEXT OR COMMENTS.
""" # noqa: E501
)
message["content"] = updated_prompt