camel-ai 0.2.72a10__py3-none-any.whl → 0.2.73a0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of camel-ai might be problematic. Click here for more details.
- camel/__init__.py +1 -1
- camel/agents/chat_agent.py +113 -338
- camel/memories/agent_memories.py +18 -17
- camel/societies/workforce/prompts.py +10 -4
- camel/societies/workforce/single_agent_worker.py +7 -5
- camel/toolkits/__init__.py +4 -1
- camel/toolkits/base.py +57 -1
- camel/toolkits/hybrid_browser_toolkit/config_loader.py +136 -413
- camel/toolkits/hybrid_browser_toolkit/hybrid_browser_toolkit.py +796 -1631
- camel/toolkits/hybrid_browser_toolkit/ts/package-lock.json +4356 -0
- camel/toolkits/hybrid_browser_toolkit/ts/package.json +33 -0
- camel/toolkits/hybrid_browser_toolkit/ts/src/browser-scripts.js +125 -0
- camel/toolkits/hybrid_browser_toolkit/ts/src/browser-session.ts +916 -0
- camel/toolkits/hybrid_browser_toolkit/ts/src/config-loader.ts +226 -0
- camel/toolkits/hybrid_browser_toolkit/ts/src/hybrid-browser-toolkit.ts +522 -0
- camel/toolkits/hybrid_browser_toolkit/ts/src/index.ts +7 -0
- camel/toolkits/hybrid_browser_toolkit/ts/src/types.ts +110 -0
- camel/toolkits/hybrid_browser_toolkit/ts/tsconfig.json +26 -0
- camel/toolkits/hybrid_browser_toolkit/ts/websocket-server.js +210 -0
- camel/toolkits/hybrid_browser_toolkit/ws_wrapper.py +533 -0
- camel/toolkits/message_integration.py +592 -0
- camel/toolkits/screenshot_toolkit.py +116 -31
- camel/toolkits/search_toolkit.py +20 -2
- camel/toolkits/terminal_toolkit.py +16 -2
- camel/toolkits/video_analysis_toolkit.py +13 -13
- camel/toolkits/video_download_toolkit.py +11 -11
- {camel_ai-0.2.72a10.dist-info → camel_ai-0.2.73a0.dist-info}/METADATA +10 -4
- {camel_ai-0.2.72a10.dist-info → camel_ai-0.2.73a0.dist-info}/RECORD +30 -24
- camel/toolkits/hybrid_browser_toolkit/actions.py +0 -417
- camel/toolkits/hybrid_browser_toolkit/agent.py +0 -311
- camel/toolkits/hybrid_browser_toolkit/browser_session.py +0 -740
- camel/toolkits/hybrid_browser_toolkit/snapshot.py +0 -227
- camel/toolkits/hybrid_browser_toolkit/stealth_script.js +0 -0
- camel/toolkits/hybrid_browser_toolkit/unified_analyzer.js +0 -1002
- {camel_ai-0.2.72a10.dist-info → camel_ai-0.2.73a0.dist-info}/WHEEL +0 -0
- {camel_ai-0.2.72a10.dist-info → camel_ai-0.2.73a0.dist-info}/licenses/LICENSE +0 -0
camel/__init__.py
CHANGED
camel/agents/chat_agent.py
CHANGED
|
@@ -74,7 +74,7 @@ from camel.models import (
|
|
|
74
74
|
from camel.prompts import TextPrompt
|
|
75
75
|
from camel.responses import ChatAgentResponse
|
|
76
76
|
from camel.storages import JsonStorage
|
|
77
|
-
from camel.toolkits import FunctionTool
|
|
77
|
+
from camel.toolkits import FunctionTool, RegisteredAgentToolkit
|
|
78
78
|
from camel.types import (
|
|
79
79
|
ChatCompletion,
|
|
80
80
|
ChatCompletionChunk,
|
|
@@ -89,7 +89,6 @@ from camel.utils import (
|
|
|
89
89
|
model_from_json_schema,
|
|
90
90
|
)
|
|
91
91
|
from camel.utils.commons import dependencies_required
|
|
92
|
-
from camel.utils.tool_result import ToolResult
|
|
93
92
|
|
|
94
93
|
if TYPE_CHECKING:
|
|
95
94
|
from camel.terminators import ResponseTerminator
|
|
@@ -348,6 +347,13 @@ class ChatAgent(BaseAgent):
|
|
|
348
347
|
tools (Optional[List[Union[FunctionTool, Callable]]], optional): List
|
|
349
348
|
of available :obj:`FunctionTool` or :obj:`Callable`. (default:
|
|
350
349
|
:obj:`None`)
|
|
350
|
+
toolkits_to_register_agent (Optional[List[RegisteredAgentToolkit]],
|
|
351
|
+
optional): List of toolkit instances that inherit from
|
|
352
|
+
:obj:`RegisteredAgentToolkit`. The agent will register itself with
|
|
353
|
+
these toolkits, allowing them to access the agent instance. Note:
|
|
354
|
+
This does NOT add the toolkit's tools to the agent. To use tools
|
|
355
|
+
from these toolkits, pass them explicitly via the `tools`
|
|
356
|
+
parameter. (default: :obj:`None`)
|
|
351
357
|
external_tools (Optional[List[Union[FunctionTool, Callable,
|
|
352
358
|
Dict[str, Any]]]], optional): List of external tools
|
|
353
359
|
(:obj:`FunctionTool` or :obj:`Callable` or :obj:`Dict[str, Any]`)
|
|
@@ -405,6 +411,9 @@ class ChatAgent(BaseAgent):
|
|
|
405
411
|
token_limit: Optional[int] = None,
|
|
406
412
|
output_language: Optional[str] = None,
|
|
407
413
|
tools: Optional[List[Union[FunctionTool, Callable]]] = None,
|
|
414
|
+
toolkits_to_register_agent: Optional[
|
|
415
|
+
List[RegisteredAgentToolkit]
|
|
416
|
+
] = None,
|
|
408
417
|
external_tools: Optional[
|
|
409
418
|
List[Union[FunctionTool, Callable, Dict[str, Any]]]
|
|
410
419
|
] = None,
|
|
@@ -438,7 +447,7 @@ class ChatAgent(BaseAgent):
|
|
|
438
447
|
token_limit or self.model_backend.token_limit,
|
|
439
448
|
)
|
|
440
449
|
|
|
441
|
-
self.
|
|
450
|
+
self._memory: AgentMemory = memory or ChatHistoryMemory(
|
|
442
451
|
context_creator,
|
|
443
452
|
window_size=message_window_size,
|
|
444
453
|
agent_id=self.agent_id,
|
|
@@ -446,7 +455,7 @@ class ChatAgent(BaseAgent):
|
|
|
446
455
|
|
|
447
456
|
# So we don't have to pass agent_id when we define memory
|
|
448
457
|
if memory is not None:
|
|
449
|
-
|
|
458
|
+
self._memory.agent_id = self.agent_id
|
|
450
459
|
|
|
451
460
|
# Set up system message and initialize messages
|
|
452
461
|
self._original_system_message = (
|
|
@@ -479,6 +488,12 @@ class ChatAgent(BaseAgent):
|
|
|
479
488
|
]
|
|
480
489
|
}
|
|
481
490
|
|
|
491
|
+
# Register agent with toolkits that have RegisteredAgentToolkit mixin
|
|
492
|
+
if toolkits_to_register_agent:
|
|
493
|
+
for toolkit in toolkits_to_register_agent:
|
|
494
|
+
if isinstance(toolkit, RegisteredAgentToolkit):
|
|
495
|
+
toolkit.register_agent(self)
|
|
496
|
+
|
|
482
497
|
self._external_tool_schemas = {
|
|
483
498
|
tool_schema["function"]["name"]: tool_schema
|
|
484
499
|
for tool_schema in [
|
|
@@ -494,9 +509,6 @@ class ChatAgent(BaseAgent):
|
|
|
494
509
|
self.tool_execution_timeout = tool_execution_timeout
|
|
495
510
|
self.mask_tool_output = mask_tool_output
|
|
496
511
|
self._secure_result_store: Dict[str, Any] = {}
|
|
497
|
-
self._pending_images: List[str] = []
|
|
498
|
-
self._image_retry_count: Dict[str, int] = {}
|
|
499
|
-
# Store images to attach to next user message
|
|
500
512
|
self.pause_event = pause_event
|
|
501
513
|
self.prune_tool_calls_from_memory = prune_tool_calls_from_memory
|
|
502
514
|
|
|
@@ -504,8 +516,6 @@ class ChatAgent(BaseAgent):
|
|
|
504
516
|
r"""Resets the :obj:`ChatAgent` to its initial state."""
|
|
505
517
|
self.terminated = False
|
|
506
518
|
self.init_messages()
|
|
507
|
-
self._pending_images = []
|
|
508
|
-
self._image_retry_count = {}
|
|
509
519
|
for terminator in self.response_terminators:
|
|
510
520
|
terminator.reset()
|
|
511
521
|
|
|
@@ -670,6 +680,25 @@ class ChatAgent(BaseAgent):
|
|
|
670
680
|
)
|
|
671
681
|
self.init_messages()
|
|
672
682
|
|
|
683
|
+
@property
|
|
684
|
+
def memory(self) -> AgentMemory:
|
|
685
|
+
r"""Returns the agent memory."""
|
|
686
|
+
return self._memory
|
|
687
|
+
|
|
688
|
+
@memory.setter
|
|
689
|
+
def memory(self, value: AgentMemory) -> None:
|
|
690
|
+
r"""Set the agent memory.
|
|
691
|
+
|
|
692
|
+
When setting a new memory, the system message is automatically
|
|
693
|
+
re-added to ensure it's not lost.
|
|
694
|
+
|
|
695
|
+
Args:
|
|
696
|
+
value (AgentMemory): The new agent memory to use.
|
|
697
|
+
"""
|
|
698
|
+
self._memory = value
|
|
699
|
+
# Ensure the new memory has the system message
|
|
700
|
+
self.init_messages()
|
|
701
|
+
|
|
673
702
|
def _get_full_tool_schemas(self) -> List[Dict[str, Any]]:
|
|
674
703
|
r"""Returns a list of tool schemas of all tools, including internal
|
|
675
704
|
and external tools.
|
|
@@ -1367,16 +1396,6 @@ class ChatAgent(BaseAgent):
|
|
|
1367
1396
|
role_name="User", content=input_message
|
|
1368
1397
|
)
|
|
1369
1398
|
|
|
1370
|
-
# Attach any pending images from previous tool calls
|
|
1371
|
-
image_list = self._process_pending_images()
|
|
1372
|
-
if image_list:
|
|
1373
|
-
# Create new message with images attached
|
|
1374
|
-
input_message = BaseMessage.make_user_message(
|
|
1375
|
-
role_name="User",
|
|
1376
|
-
content=input_message.content,
|
|
1377
|
-
image_list=image_list,
|
|
1378
|
-
)
|
|
1379
|
-
|
|
1380
1399
|
# Add user input to memory
|
|
1381
1400
|
self.update_memory(input_message, OpenAIBackendRole.USER)
|
|
1382
1401
|
|
|
@@ -1571,16 +1590,6 @@ class ChatAgent(BaseAgent):
|
|
|
1571
1590
|
role_name="User", content=input_message
|
|
1572
1591
|
)
|
|
1573
1592
|
|
|
1574
|
-
# Attach any pending images from previous tool calls
|
|
1575
|
-
image_list = self._process_pending_images()
|
|
1576
|
-
if image_list:
|
|
1577
|
-
# Create new message with images attached
|
|
1578
|
-
input_message = BaseMessage.make_user_message(
|
|
1579
|
-
role_name="User",
|
|
1580
|
-
content=input_message.content,
|
|
1581
|
-
image_list=image_list,
|
|
1582
|
-
)
|
|
1583
|
-
|
|
1584
1593
|
self.update_memory(input_message, OpenAIBackendRole.USER)
|
|
1585
1594
|
|
|
1586
1595
|
tool_call_records: List[ToolCallingRecord] = []
|
|
@@ -1634,7 +1643,6 @@ class ChatAgent(BaseAgent):
|
|
|
1634
1643
|
|
|
1635
1644
|
if tool_call_requests := response.tool_call_requests:
|
|
1636
1645
|
# Process all tool calls
|
|
1637
|
-
new_images_from_tools = []
|
|
1638
1646
|
for tool_call_request in tool_call_requests:
|
|
1639
1647
|
if (
|
|
1640
1648
|
tool_call_request.tool_name
|
|
@@ -1654,72 +1662,10 @@ class ChatAgent(BaseAgent):
|
|
|
1654
1662
|
)
|
|
1655
1663
|
tool_call_records.append(tool_call_record)
|
|
1656
1664
|
|
|
1657
|
-
# Check if this tool call produced images
|
|
1658
|
-
if (
|
|
1659
|
-
hasattr(tool_call_record, 'images')
|
|
1660
|
-
and tool_call_record.images
|
|
1661
|
-
):
|
|
1662
|
-
new_images_from_tools.extend(
|
|
1663
|
-
tool_call_record.images
|
|
1664
|
-
)
|
|
1665
|
-
|
|
1666
1665
|
# If we found an external tool call, break the loop
|
|
1667
1666
|
if external_tool_call_requests:
|
|
1668
1667
|
break
|
|
1669
1668
|
|
|
1670
|
-
# If tools produced images
|
|
1671
|
-
# send them to the model as a user message
|
|
1672
|
-
if new_images_from_tools:
|
|
1673
|
-
# Convert base64 images to PIL Images
|
|
1674
|
-
image_list = []
|
|
1675
|
-
for img_data in new_images_from_tools:
|
|
1676
|
-
try:
|
|
1677
|
-
import base64
|
|
1678
|
-
import io
|
|
1679
|
-
|
|
1680
|
-
from PIL import Image
|
|
1681
|
-
|
|
1682
|
-
# Extract base64 data from data URL format
|
|
1683
|
-
if img_data.startswith("data:image"):
|
|
1684
|
-
# Format:
|
|
1685
|
-
# "..."
|
|
1686
|
-
base64_data = img_data.split(',', 1)[1]
|
|
1687
|
-
else:
|
|
1688
|
-
# Raw base64 data
|
|
1689
|
-
base64_data = img_data
|
|
1690
|
-
|
|
1691
|
-
# Decode and create PIL Image
|
|
1692
|
-
image_bytes = base64.b64decode(base64_data)
|
|
1693
|
-
pil_image = Image.open(io.BytesIO(image_bytes))
|
|
1694
|
-
# Convert to ensure proper
|
|
1695
|
-
# Image.Image type for compatibility
|
|
1696
|
-
pil_image_tool_result: Image.Image = (
|
|
1697
|
-
pil_image.convert('RGB')
|
|
1698
|
-
)
|
|
1699
|
-
image_list.append(pil_image_tool_result)
|
|
1700
|
-
|
|
1701
|
-
except Exception as e:
|
|
1702
|
-
logger.warning(
|
|
1703
|
-
f"Failed to convert "
|
|
1704
|
-
f"base64 image to PIL for immediate use: {e}"
|
|
1705
|
-
)
|
|
1706
|
-
continue
|
|
1707
|
-
|
|
1708
|
-
# If we have valid images
|
|
1709
|
-
# create a user message with images
|
|
1710
|
-
if image_list:
|
|
1711
|
-
# Create a user message with images
|
|
1712
|
-
# to provide visual context immediately
|
|
1713
|
-
image_message = BaseMessage.make_user_message(
|
|
1714
|
-
role_name="User",
|
|
1715
|
-
content="[Visual content from tool execution - please analyze and continue]", # noqa: E501
|
|
1716
|
-
image_list=image_list,
|
|
1717
|
-
)
|
|
1718
|
-
|
|
1719
|
-
self.update_memory(
|
|
1720
|
-
image_message, OpenAIBackendRole.USER
|
|
1721
|
-
)
|
|
1722
|
-
|
|
1723
1669
|
if (
|
|
1724
1670
|
self.max_iteration is not None
|
|
1725
1671
|
and iteration_count >= self.max_iteration
|
|
@@ -1810,69 +1756,6 @@ class ChatAgent(BaseAgent):
|
|
|
1810
1756
|
info=info,
|
|
1811
1757
|
)
|
|
1812
1758
|
|
|
1813
|
-
def _process_pending_images(self) -> List:
|
|
1814
|
-
r"""Process pending images with retry logic and return PIL Image list.
|
|
1815
|
-
|
|
1816
|
-
Returns:
|
|
1817
|
-
List: List of successfully converted PIL Images.
|
|
1818
|
-
"""
|
|
1819
|
-
if not self._pending_images:
|
|
1820
|
-
return []
|
|
1821
|
-
|
|
1822
|
-
image_list = []
|
|
1823
|
-
successfully_processed = []
|
|
1824
|
-
failed_images = []
|
|
1825
|
-
|
|
1826
|
-
for img_data in self._pending_images:
|
|
1827
|
-
# Track retry count
|
|
1828
|
-
retry_count = self._image_retry_count.get(img_data, 0)
|
|
1829
|
-
|
|
1830
|
-
# Remove images that have failed too many times (max 3 attempts)
|
|
1831
|
-
if retry_count >= 3:
|
|
1832
|
-
failed_images.append(img_data)
|
|
1833
|
-
logger.warning(
|
|
1834
|
-
f"Removing image after {retry_count} failed attempts"
|
|
1835
|
-
)
|
|
1836
|
-
continue
|
|
1837
|
-
|
|
1838
|
-
try:
|
|
1839
|
-
import base64
|
|
1840
|
-
import io
|
|
1841
|
-
|
|
1842
|
-
from PIL import Image
|
|
1843
|
-
|
|
1844
|
-
# Extract base64 data from data URL format
|
|
1845
|
-
if img_data.startswith("data:image"):
|
|
1846
|
-
# Format: "..."
|
|
1847
|
-
base64_data = img_data.split(',', 1)[1]
|
|
1848
|
-
else:
|
|
1849
|
-
# Raw base64 data
|
|
1850
|
-
base64_data = img_data
|
|
1851
|
-
|
|
1852
|
-
# Decode and create PIL Image
|
|
1853
|
-
image_bytes = base64.b64decode(base64_data)
|
|
1854
|
-
pil_image = Image.open(io.BytesIO(image_bytes))
|
|
1855
|
-
pil_image_converted: Image.Image = pil_image.convert('RGB')
|
|
1856
|
-
image_list.append(pil_image_converted)
|
|
1857
|
-
successfully_processed.append(img_data)
|
|
1858
|
-
|
|
1859
|
-
except Exception as e:
|
|
1860
|
-
# Increment retry count for failed conversion
|
|
1861
|
-
self._image_retry_count[img_data] = retry_count + 1
|
|
1862
|
-
logger.warning(
|
|
1863
|
-
f"Failed to convert base64 image to PIL "
|
|
1864
|
-
f"(attempt {retry_count + 1}/3): {e}"
|
|
1865
|
-
)
|
|
1866
|
-
continue
|
|
1867
|
-
|
|
1868
|
-
# Clean up processed and failed images
|
|
1869
|
-
for img in successfully_processed + failed_images:
|
|
1870
|
-
self._pending_images.remove(img)
|
|
1871
|
-
# Clean up retry count for processed/removed images
|
|
1872
|
-
self._image_retry_count.pop(img, None)
|
|
1873
|
-
|
|
1874
|
-
return image_list
|
|
1875
|
-
|
|
1876
1759
|
def _record_final_output(self, output_messages: List[BaseMessage]) -> None:
|
|
1877
1760
|
r"""Log final messages or warnings about multiple responses."""
|
|
1878
1761
|
if len(output_messages) == 1:
|
|
@@ -1883,61 +1766,6 @@ class ChatAgent(BaseAgent):
|
|
|
1883
1766
|
"selected message manually using `record_message()`."
|
|
1884
1767
|
)
|
|
1885
1768
|
|
|
1886
|
-
def _is_vision_error(self, exc: Exception) -> bool:
|
|
1887
|
-
r"""Check if the exception is likely related to vision/image is not
|
|
1888
|
-
supported by the model."""
|
|
1889
|
-
# TODO: more robust vision error detection
|
|
1890
|
-
error_msg = str(exc).lower()
|
|
1891
|
-
vision_keywords = [
|
|
1892
|
-
'vision',
|
|
1893
|
-
'image',
|
|
1894
|
-
'multimodal',
|
|
1895
|
-
'unsupported',
|
|
1896
|
-
'invalid content type',
|
|
1897
|
-
'image_url',
|
|
1898
|
-
'visual',
|
|
1899
|
-
]
|
|
1900
|
-
return any(keyword in error_msg for keyword in vision_keywords)
|
|
1901
|
-
|
|
1902
|
-
def _has_images(self, messages: List[OpenAIMessage]) -> bool:
|
|
1903
|
-
r"""Check if any message contains images."""
|
|
1904
|
-
for msg in messages:
|
|
1905
|
-
content = msg.get('content')
|
|
1906
|
-
if isinstance(content, list):
|
|
1907
|
-
for item in content:
|
|
1908
|
-
if (
|
|
1909
|
-
isinstance(item, dict)
|
|
1910
|
-
and item.get('type') == 'image_url'
|
|
1911
|
-
):
|
|
1912
|
-
return True
|
|
1913
|
-
return False
|
|
1914
|
-
|
|
1915
|
-
def _strip_images_from_messages(
|
|
1916
|
-
self, messages: List[OpenAIMessage]
|
|
1917
|
-
) -> List[OpenAIMessage]:
|
|
1918
|
-
r"""Remove images from messages, keeping only text content."""
|
|
1919
|
-
stripped_messages = []
|
|
1920
|
-
for msg in messages:
|
|
1921
|
-
content = msg.get('content')
|
|
1922
|
-
if isinstance(content, list):
|
|
1923
|
-
# Extract only text content from multimodal messages
|
|
1924
|
-
text_content = ""
|
|
1925
|
-
for item in content:
|
|
1926
|
-
if isinstance(item, dict) and item.get('type') == 'text':
|
|
1927
|
-
text_content += item.get('text', '')
|
|
1928
|
-
|
|
1929
|
-
# Create new message with only text content
|
|
1930
|
-
new_msg = msg.copy()
|
|
1931
|
-
new_msg['content'] = (
|
|
1932
|
-
text_content
|
|
1933
|
-
or "[Image content removed - model doesn't support vision]"
|
|
1934
|
-
)
|
|
1935
|
-
stripped_messages.append(new_msg)
|
|
1936
|
-
else:
|
|
1937
|
-
# Regular text message, keep as is
|
|
1938
|
-
stripped_messages.append(msg)
|
|
1939
|
-
return stripped_messages
|
|
1940
|
-
|
|
1941
1769
|
@observe()
|
|
1942
1770
|
def _get_model_response(
|
|
1943
1771
|
self,
|
|
@@ -1971,35 +1799,13 @@ class ChatAgent(BaseAgent):
|
|
|
1971
1799
|
openai_messages, response_format, tool_schemas or None
|
|
1972
1800
|
)
|
|
1973
1801
|
except Exception as exc:
|
|
1974
|
-
|
|
1975
|
-
|
|
1976
|
-
|
|
1977
|
-
|
|
1978
|
-
|
|
1979
|
-
|
|
1980
|
-
|
|
1981
|
-
)
|
|
1982
|
-
try:
|
|
1983
|
-
stripped_messages = self._strip_images_from_messages(
|
|
1984
|
-
openai_messages
|
|
1985
|
-
)
|
|
1986
|
-
response = self.model_backend.run(
|
|
1987
|
-
stripped_messages,
|
|
1988
|
-
response_format,
|
|
1989
|
-
tool_schemas or None,
|
|
1990
|
-
)
|
|
1991
|
-
except Exception:
|
|
1992
|
-
pass # Fall through to original error handling
|
|
1993
|
-
|
|
1994
|
-
if not response:
|
|
1995
|
-
logger.error(
|
|
1996
|
-
f"An error occurred while running model "
|
|
1997
|
-
f"iteration {current_iteration}, "
|
|
1998
|
-
f"{self.model_backend.model_type}, "
|
|
1999
|
-
f"index: {self.model_backend.current_model_index}",
|
|
2000
|
-
exc_info=exc,
|
|
2001
|
-
)
|
|
2002
|
-
error_info = str(exc)
|
|
1802
|
+
logger.error(
|
|
1803
|
+
f"An error occurred while running model "
|
|
1804
|
+
f"{self.model_backend.model_type}, "
|
|
1805
|
+
f"index: {self.model_backend.current_model_index}",
|
|
1806
|
+
exc_info=exc,
|
|
1807
|
+
)
|
|
1808
|
+
error_info = str(exc)
|
|
2003
1809
|
|
|
2004
1810
|
if not response and self.model_backend.num_models > 1:
|
|
2005
1811
|
raise ModelProcessingError(
|
|
@@ -2060,33 +1866,13 @@ class ChatAgent(BaseAgent):
|
|
|
2060
1866
|
openai_messages, response_format, tool_schemas or None
|
|
2061
1867
|
)
|
|
2062
1868
|
except Exception as exc:
|
|
2063
|
-
|
|
2064
|
-
|
|
2065
|
-
|
|
2066
|
-
|
|
2067
|
-
|
|
2068
|
-
|
|
2069
|
-
|
|
2070
|
-
try:
|
|
2071
|
-
stripped_messages = self._strip_images_from_messages(
|
|
2072
|
-
openai_messages
|
|
2073
|
-
)
|
|
2074
|
-
response = await self.model_backend.arun(
|
|
2075
|
-
stripped_messages,
|
|
2076
|
-
response_format,
|
|
2077
|
-
tool_schemas or None,
|
|
2078
|
-
)
|
|
2079
|
-
except Exception:
|
|
2080
|
-
pass # Fall through to original error handling
|
|
2081
|
-
|
|
2082
|
-
if not response:
|
|
2083
|
-
logger.error(
|
|
2084
|
-
f"An error occurred while running model "
|
|
2085
|
-
f"{self.model_backend.model_type}, "
|
|
2086
|
-
f"index: {self.model_backend.current_model_index}",
|
|
2087
|
-
exc_info=exc,
|
|
2088
|
-
)
|
|
2089
|
-
error_info = str(exc)
|
|
1869
|
+
logger.error(
|
|
1870
|
+
f"An error occurred while running model "
|
|
1871
|
+
f"{self.model_backend.model_type}, "
|
|
1872
|
+
f"index: {self.model_backend.current_model_index}",
|
|
1873
|
+
exc_info=exc,
|
|
1874
|
+
)
|
|
1875
|
+
error_info = str(exc)
|
|
2090
1876
|
|
|
2091
1877
|
if not response and self.model_backend.num_models > 1:
|
|
2092
1878
|
raise ModelProcessingError(
|
|
@@ -2457,26 +2243,9 @@ class ChatAgent(BaseAgent):
|
|
|
2457
2243
|
mask_flag = False
|
|
2458
2244
|
logger.warning(f"{error_msg} with result: {result}")
|
|
2459
2245
|
|
|
2460
|
-
|
|
2461
|
-
images_to_attach = None
|
|
2462
|
-
if isinstance(result, ToolResult):
|
|
2463
|
-
images_to_attach = result.images
|
|
2464
|
-
result = str(result) # Use string representation for storage
|
|
2465
|
-
|
|
2466
|
-
tool_record = self._record_tool_calling(
|
|
2246
|
+
return self._record_tool_calling(
|
|
2467
2247
|
func_name, args, result, tool_call_id, mask_output=mask_flag
|
|
2468
2248
|
)
|
|
2469
|
-
logger.info(f"Tool calling record:\n{tool_record}")
|
|
2470
|
-
|
|
2471
|
-
# Store images for later attachment to next user message
|
|
2472
|
-
if images_to_attach:
|
|
2473
|
-
tool_record.images = images_to_attach
|
|
2474
|
-
# Add images with duplicate prevention
|
|
2475
|
-
for img in images_to_attach:
|
|
2476
|
-
if img not in self._pending_images:
|
|
2477
|
-
self._pending_images.append(img)
|
|
2478
|
-
|
|
2479
|
-
return tool_record
|
|
2480
2249
|
|
|
2481
2250
|
async def _aexecute_tool(
|
|
2482
2251
|
self,
|
|
@@ -2517,26 +2286,7 @@ class ChatAgent(BaseAgent):
|
|
|
2517
2286
|
error_msg = f"Error executing async tool '{func_name}': {e!s}"
|
|
2518
2287
|
result = f"Tool execution failed: {error_msg}"
|
|
2519
2288
|
logging.warning(error_msg)
|
|
2520
|
-
|
|
2521
|
-
# Check if result is a ToolResult with images
|
|
2522
|
-
images_to_attach = None
|
|
2523
|
-
if isinstance(result, ToolResult):
|
|
2524
|
-
images_to_attach = result.images
|
|
2525
|
-
result = str(result) # Use string representation for storage
|
|
2526
|
-
|
|
2527
|
-
tool_record = self._record_tool_calling(
|
|
2528
|
-
func_name, args, result, tool_call_id
|
|
2529
|
-
)
|
|
2530
|
-
|
|
2531
|
-
# Store images for later attachment to next user message
|
|
2532
|
-
if images_to_attach:
|
|
2533
|
-
tool_record.images = images_to_attach
|
|
2534
|
-
# Add images with duplicate prevention
|
|
2535
|
-
for img in images_to_attach:
|
|
2536
|
-
if img not in self._pending_images:
|
|
2537
|
-
self._pending_images.append(img)
|
|
2538
|
-
|
|
2539
|
-
return tool_record
|
|
2289
|
+
return self._record_tool_calling(func_name, args, result, tool_call_id)
|
|
2540
2290
|
|
|
2541
2291
|
def _record_tool_calling(
|
|
2542
2292
|
self,
|
|
@@ -3802,6 +3552,9 @@ class ChatAgent(BaseAgent):
|
|
|
3802
3552
|
# To avoid duplicated system memory.
|
|
3803
3553
|
system_message = None if with_memory else self._original_system_message
|
|
3804
3554
|
|
|
3555
|
+
# Clone tools and collect toolkits that need registration
|
|
3556
|
+
cloned_tools, toolkits_to_register = self._clone_tools()
|
|
3557
|
+
|
|
3805
3558
|
new_agent = ChatAgent(
|
|
3806
3559
|
system_message=system_message,
|
|
3807
3560
|
model=self.model_backend.models, # Pass the existing model_backend
|
|
@@ -3811,7 +3564,8 @@ class ChatAgent(BaseAgent):
|
|
|
3811
3564
|
self.memory.get_context_creator(), "token_limit", None
|
|
3812
3565
|
),
|
|
3813
3566
|
output_language=self._output_language,
|
|
3814
|
-
tools=
|
|
3567
|
+
tools=cloned_tools,
|
|
3568
|
+
toolkits_to_register_agent=toolkits_to_register,
|
|
3815
3569
|
external_tools=[
|
|
3816
3570
|
schema for schema in self._external_tool_schemas.values()
|
|
3817
3571
|
],
|
|
@@ -3836,55 +3590,76 @@ class ChatAgent(BaseAgent):
|
|
|
3836
3590
|
|
|
3837
3591
|
return new_agent
|
|
3838
3592
|
|
|
3839
|
-
def _clone_tools(
|
|
3840
|
-
|
|
3841
|
-
|
|
3593
|
+
def _clone_tools(
|
|
3594
|
+
self,
|
|
3595
|
+
) -> Tuple[
|
|
3596
|
+
List[Union[FunctionTool, Callable]], List[RegisteredAgentToolkit]
|
|
3597
|
+
]:
|
|
3598
|
+
r"""Clone tools and return toolkits that need agent registration.
|
|
3599
|
+
|
|
3600
|
+
This method handles stateful toolkits by cloning them if they have
|
|
3601
|
+
a clone_for_new_session method, and collecting RegisteredAgentToolkit
|
|
3602
|
+
instances for later registration.
|
|
3603
|
+
|
|
3604
|
+
Returns:
|
|
3605
|
+
Tuple containing:
|
|
3606
|
+
- List of cloned tools/functions
|
|
3607
|
+
- List of RegisteredAgentToolkit instances need registration
|
|
3608
|
+
"""
|
|
3842
3609
|
cloned_tools = []
|
|
3843
|
-
|
|
3610
|
+
toolkits_to_register = []
|
|
3611
|
+
cloned_toolkits = {}
|
|
3612
|
+
# Cache for cloned toolkits by original toolkit id
|
|
3844
3613
|
|
|
3845
3614
|
for tool in self._internal_tools.values():
|
|
3846
|
-
# Check if this is a
|
|
3847
|
-
if (
|
|
3848
|
-
hasattr(tool.func, '__self__')
|
|
3849
|
-
and tool.func.__self__.__class__.__name__
|
|
3850
|
-
== 'HybridBrowserToolkit'
|
|
3851
|
-
):
|
|
3615
|
+
# Check if this tool is a method bound to a toolkit instance
|
|
3616
|
+
if hasattr(tool.func, '__self__'):
|
|
3852
3617
|
toolkit_instance = tool.func.__self__
|
|
3853
3618
|
toolkit_id = id(toolkit_instance)
|
|
3854
3619
|
|
|
3855
|
-
|
|
3856
|
-
|
|
3857
|
-
|
|
3858
|
-
|
|
3620
|
+
if toolkit_id not in cloned_toolkits:
|
|
3621
|
+
# Check if the toolkit has a clone method
|
|
3622
|
+
if hasattr(toolkit_instance, 'clone_for_new_session'):
|
|
3623
|
+
try:
|
|
3624
|
+
import uuid
|
|
3859
3625
|
|
|
3860
|
-
|
|
3861
|
-
|
|
3862
|
-
|
|
3863
|
-
|
|
3864
|
-
|
|
3865
|
-
|
|
3866
|
-
|
|
3867
|
-
|
|
3868
|
-
|
|
3869
|
-
|
|
3870
|
-
|
|
3871
|
-
|
|
3626
|
+
new_session_id = str(uuid.uuid4())[:8]
|
|
3627
|
+
new_toolkit = (
|
|
3628
|
+
toolkit_instance.clone_for_new_session(
|
|
3629
|
+
new_session_id
|
|
3630
|
+
)
|
|
3631
|
+
)
|
|
3632
|
+
|
|
3633
|
+
# If this is a RegisteredAgentToolkit,
|
|
3634
|
+
# add it to registration list
|
|
3635
|
+
if isinstance(new_toolkit, RegisteredAgentToolkit):
|
|
3636
|
+
toolkits_to_register.append(new_toolkit)
|
|
3637
|
+
|
|
3638
|
+
cloned_toolkits[toolkit_id] = new_toolkit
|
|
3639
|
+
except Exception as e:
|
|
3640
|
+
logger.warning(
|
|
3641
|
+
f"Failed to clone toolkit {toolkit_instance.__class__.__name__}: {e}" # noqa:E501
|
|
3642
|
+
)
|
|
3643
|
+
# Use original toolkit if cloning fails
|
|
3644
|
+
cloned_toolkits[toolkit_id] = toolkit_instance
|
|
3645
|
+
else:
|
|
3646
|
+
# Toolkit doesn't support cloning, use original
|
|
3647
|
+
cloned_toolkits[toolkit_id] = toolkit_instance
|
|
3872
3648
|
|
|
3873
|
-
# Get the
|
|
3874
|
-
|
|
3649
|
+
# Get the method from the cloned (or original) toolkit
|
|
3650
|
+
toolkit = cloned_toolkits[toolkit_id]
|
|
3875
3651
|
method_name = tool.func.__name__
|
|
3876
|
-
if hasattr(
|
|
3877
|
-
new_method = getattr(
|
|
3652
|
+
if hasattr(toolkit, method_name):
|
|
3653
|
+
new_method = getattr(toolkit, method_name)
|
|
3878
3654
|
cloned_tools.append(new_method)
|
|
3879
3655
|
else:
|
|
3880
3656
|
# Fallback to original function
|
|
3881
3657
|
cloned_tools.append(tool.func)
|
|
3882
3658
|
else:
|
|
3883
|
-
#
|
|
3884
|
-
# just use the original function
|
|
3659
|
+
# Not a toolkit method, just use the original function
|
|
3885
3660
|
cloned_tools.append(tool.func)
|
|
3886
3661
|
|
|
3887
|
-
return cloned_tools
|
|
3662
|
+
return cloned_tools, toolkits_to_register
|
|
3888
3663
|
|
|
3889
3664
|
def __repr__(self) -> str:
|
|
3890
3665
|
r"""Returns a string representation of the :obj:`ChatAgent`.
|
camel/memories/agent_memories.py
CHANGED
|
@@ -101,30 +101,31 @@ class ChatHistoryMemory(AgentMemory):
|
|
|
101
101
|
if not record_dicts:
|
|
102
102
|
return
|
|
103
103
|
|
|
104
|
-
#
|
|
105
|
-
|
|
106
|
-
|
|
104
|
+
# Track indices to remove (reverse order for efficient deletion)
|
|
105
|
+
indices_to_remove = []
|
|
106
|
+
|
|
107
|
+
# Identify indices of tool-related messages
|
|
108
|
+
for i, record in enumerate(record_dicts):
|
|
107
109
|
role = record.get('role_at_backend')
|
|
108
110
|
|
|
109
|
-
#
|
|
111
|
+
# Mark FUNCTION messages for removal
|
|
110
112
|
if role == OpenAIBackendRole.FUNCTION.value:
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
# Skip ASSISTANT messages with tool_calls
|
|
118
|
-
if role == OpenAIBackendRole.ASSISTANT.value:
|
|
113
|
+
indices_to_remove.append(i)
|
|
114
|
+
# Mark TOOL messages for removal
|
|
115
|
+
elif role == OpenAIBackendRole.TOOL.value:
|
|
116
|
+
indices_to_remove.append(i)
|
|
117
|
+
# Mark ASSISTANT messages with tool_calls for removal
|
|
118
|
+
elif role == OpenAIBackendRole.ASSISTANT.value:
|
|
119
119
|
meta_dict = record.get('meta_dict', {})
|
|
120
120
|
if meta_dict and 'tool_calls' in meta_dict:
|
|
121
|
-
|
|
121
|
+
indices_to_remove.append(i)
|
|
122
122
|
|
|
123
|
-
|
|
124
|
-
|
|
123
|
+
# Remove records in-place
|
|
124
|
+
for i in reversed(indices_to_remove):
|
|
125
|
+
del record_dicts[i]
|
|
125
126
|
|
|
126
|
-
# Save the
|
|
127
|
-
self._chat_history_block.storage.save(
|
|
127
|
+
# Save the modified records back to storage
|
|
128
|
+
self._chat_history_block.storage.save(record_dicts)
|
|
128
129
|
|
|
129
130
|
|
|
130
131
|
class VectorDBMemory(AgentMemory):
|