aient 1.2.14__tar.gz → 1.2.16__tar.gz

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.
Files changed (56) hide show
  1. {aient-1.2.14 → aient-1.2.16}/PKG-INFO +1 -1
  2. {aient-1.2.14 → aient-1.2.16}/aient/architext/architext/core.py +17 -10
  3. {aient-1.2.14 → aient-1.2.16}/aient/architext/test/test.py +88 -4
  4. {aient-1.2.14 → aient-1.2.16}/aient.egg-info/PKG-INFO +1 -1
  5. {aient-1.2.14 → aient-1.2.16}/pyproject.toml +1 -1
  6. {aient-1.2.14 → aient-1.2.16}/LICENSE +0 -0
  7. {aient-1.2.14 → aient-1.2.16}/README.md +0 -0
  8. {aient-1.2.14 → aient-1.2.16}/aient/__init__.py +0 -0
  9. {aient-1.2.14 → aient-1.2.16}/aient/architext/architext/__init__.py +0 -0
  10. {aient-1.2.14 → aient-1.2.16}/aient/architext/test/openai_client.py +0 -0
  11. {aient-1.2.14 → aient-1.2.16}/aient/architext/test/test_save_load.py +0 -0
  12. {aient-1.2.14 → aient-1.2.16}/aient/core/__init__.py +0 -0
  13. {aient-1.2.14 → aient-1.2.16}/aient/core/log_config.py +0 -0
  14. {aient-1.2.14 → aient-1.2.16}/aient/core/models.py +0 -0
  15. {aient-1.2.14 → aient-1.2.16}/aient/core/request.py +0 -0
  16. {aient-1.2.14 → aient-1.2.16}/aient/core/response.py +0 -0
  17. {aient-1.2.14 → aient-1.2.16}/aient/core/test/test_base_api.py +0 -0
  18. {aient-1.2.14 → aient-1.2.16}/aient/core/test/test_geminimask.py +0 -0
  19. {aient-1.2.14 → aient-1.2.16}/aient/core/test/test_image.py +0 -0
  20. {aient-1.2.14 → aient-1.2.16}/aient/core/test/test_payload.py +0 -0
  21. {aient-1.2.14 → aient-1.2.16}/aient/core/utils.py +0 -0
  22. {aient-1.2.14 → aient-1.2.16}/aient/models/__init__.py +0 -0
  23. {aient-1.2.14 → aient-1.2.16}/aient/models/audio.py +0 -0
  24. {aient-1.2.14 → aient-1.2.16}/aient/models/base.py +0 -0
  25. {aient-1.2.14 → aient-1.2.16}/aient/models/chatgpt.py +0 -0
  26. {aient-1.2.14 → aient-1.2.16}/aient/plugins/__init__.py +0 -0
  27. {aient-1.2.14 → aient-1.2.16}/aient/plugins/arXiv.py +0 -0
  28. {aient-1.2.14 → aient-1.2.16}/aient/plugins/config.py +0 -0
  29. {aient-1.2.14 → aient-1.2.16}/aient/plugins/excute_command.py +0 -0
  30. {aient-1.2.14 → aient-1.2.16}/aient/plugins/get_time.py +0 -0
  31. {aient-1.2.14 → aient-1.2.16}/aient/plugins/image.py +0 -0
  32. {aient-1.2.14 → aient-1.2.16}/aient/plugins/list_directory.py +0 -0
  33. {aient-1.2.14 → aient-1.2.16}/aient/plugins/read_file.py +0 -0
  34. {aient-1.2.14 → aient-1.2.16}/aient/plugins/read_image.py +0 -0
  35. {aient-1.2.14 → aient-1.2.16}/aient/plugins/readonly.py +0 -0
  36. {aient-1.2.14 → aient-1.2.16}/aient/plugins/registry.py +0 -0
  37. {aient-1.2.14 → aient-1.2.16}/aient/plugins/run_python.py +0 -0
  38. {aient-1.2.14 → aient-1.2.16}/aient/plugins/websearch.py +0 -0
  39. {aient-1.2.14 → aient-1.2.16}/aient/plugins/write_file.py +0 -0
  40. {aient-1.2.14 → aient-1.2.16}/aient/utils/__init__.py +0 -0
  41. {aient-1.2.14 → aient-1.2.16}/aient/utils/prompt.py +0 -0
  42. {aient-1.2.14 → aient-1.2.16}/aient/utils/scripts.py +0 -0
  43. {aient-1.2.14 → aient-1.2.16}/aient.egg-info/SOURCES.txt +0 -0
  44. {aient-1.2.14 → aient-1.2.16}/aient.egg-info/dependency_links.txt +0 -0
  45. {aient-1.2.14 → aient-1.2.16}/aient.egg-info/requires.txt +0 -0
  46. {aient-1.2.14 → aient-1.2.16}/aient.egg-info/top_level.txt +0 -0
  47. {aient-1.2.14 → aient-1.2.16}/setup.cfg +0 -0
  48. {aient-1.2.14 → aient-1.2.16}/test/test_Web_crawler.py +0 -0
  49. {aient-1.2.14 → aient-1.2.16}/test/test_ddg_search.py +0 -0
  50. {aient-1.2.14 → aient-1.2.16}/test/test_google_search.py +0 -0
  51. {aient-1.2.14 → aient-1.2.16}/test/test_ollama.py +0 -0
  52. {aient-1.2.14 → aient-1.2.16}/test/test_plugin.py +0 -0
  53. {aient-1.2.14 → aient-1.2.16}/test/test_search.py +0 -0
  54. {aient-1.2.14 → aient-1.2.16}/test/test_url.py +0 -0
  55. {aient-1.2.14 → aient-1.2.16}/test/test_whisper.py +0 -0
  56. {aient-1.2.14 → aient-1.2.16}/test/test_yjh.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: aient
3
- Version: 1.2.14
3
+ Version: 1.2.16
4
4
  Summary: Aient: The Awakening of Agent.
5
5
  Requires-Python: >=3.11
6
6
  Description-Content-Type: text/markdown
@@ -58,11 +58,11 @@ class ContentBlock:
58
58
 
59
59
  # 2. 上下文提供者 (带缓存)
60
60
  class ContextProvider(ABC):
61
- def __init__(self, name: str):
61
+ def __init__(self, name: str, visible: bool = True):
62
62
  self.name = name
63
63
  self._cached_content: Optional[str] = None
64
64
  self._is_stale: bool = True
65
- self._visible: bool = True
65
+ self._visible: bool = visible
66
66
 
67
67
  def __str__(self):
68
68
  # This allows the object to be captured when used inside an f-string.
@@ -96,8 +96,15 @@ class ContextProvider(ABC):
96
96
  return ContentBlock(self.name, self._cached_content)
97
97
  return None
98
98
 
99
+ def __add__(self, other):
100
+ if isinstance(other, Message):
101
+ # Create a new message of the same type as `other`, with `self` prepended.
102
+ new_items = [self] + other.provider()
103
+ return type(other)(*new_items)
104
+ return NotImplemented
105
+
99
106
  class Texts(ContextProvider):
100
- def __init__(self, text: Optional[Union[str, Callable[[], str]]] = None, name: Optional[str] = None):
107
+ def __init__(self, text: Optional[Union[str, Callable[[], str]]] = None, name: Optional[str] = None, visible: bool = True):
101
108
  if text is None and name is None:
102
109
  raise ValueError("Either 'text' or 'name' must be provided.")
103
110
 
@@ -119,7 +126,7 @@ class Texts(ContextProvider):
119
126
  _name = f"text_{h[:8]}"
120
127
  else:
121
128
  _name = name
122
- super().__init__(_name)
129
+ super().__init__(_name, visible=visible)
123
130
 
124
131
  async def refresh(self):
125
132
  if self._is_dynamic:
@@ -180,8 +187,8 @@ class Texts(ContextProvider):
180
187
  return self.content == other.content
181
188
 
182
189
  class Tools(ContextProvider):
183
- def __init__(self, tools_json: Optional[List[Dict]] = None, name: str = "tools"):
184
- super().__init__(name)
190
+ def __init__(self, tools_json: Optional[List[Dict]] = None, name: str = "tools", visible: bool = True):
191
+ super().__init__(name, visible=visible)
185
192
  self._tools_json = tools_json or []
186
193
  def update(self, tools_json: List[Dict]):
187
194
  self._tools_json = tools_json
@@ -197,8 +204,8 @@ class Tools(ContextProvider):
197
204
  return self._tools_json == other._tools_json
198
205
 
199
206
  class Files(ContextProvider):
200
- def __init__(self, *paths: Union[str, List[str]], name: str = "files"):
201
- super().__init__(name)
207
+ def __init__(self, *paths: Union[str, List[str]], name: str = "files", visible: bool = True):
208
+ super().__init__(name, visible=visible)
202
209
  self._files: Dict[str, str] = {}
203
210
 
204
211
  file_paths: List[str] = []
@@ -278,8 +285,8 @@ class Files(ContextProvider):
278
285
  return self._files == other._files
279
286
 
280
287
  class Images(ContextProvider):
281
- def __init__(self, url: str, name: Optional[str] = None):
282
- super().__init__(name or url)
288
+ def __init__(self, url: str, name: Optional[str] = None, visible: bool = True):
289
+ super().__init__(name or url, visible=visible)
283
290
  self.url = url
284
291
  def update(self, url: str):
285
292
  self.url = url
@@ -1158,7 +1158,46 @@ Current time: {Texts(lambda: datetime.now().strftime("%Y-%m-%d %H:%M:%S"))}
1158
1158
  with self.assertRaises(IndexError):
1159
1159
  _ = mess[2]
1160
1160
 
1161
- async def test_zb_message_provider_by_name(self):
1161
+ async def test_zb_fstring_provider_invisible_on_init(self):
1162
+ """测试在f-string中初始化的provider可以被设置为不可见"""
1163
+
1164
+ # 1. 在 f-string 中初始化一个 provider 并设置 visible=False
1165
+ # 在修改前,这会因为 __init__ 不接受 'visible' 参数而失败
1166
+ message_with_invisible_provider = f"""
1167
+ Tools: {Tools(tools_json=[{"name": "should_not_appear"}], visible=False)}
1168
+ Files: {Files(visible=True, name="files")}
1169
+ """
1170
+
1171
+ messages = Messages(UserMessage(message_with_invisible_provider))
1172
+
1173
+ # 2. 准备 Files provider 的内容
1174
+ test_file = "test_invisible_fstring.txt"
1175
+ with open(test_file, "w") as f:
1176
+ f.write("visible content")
1177
+
1178
+ try:
1179
+ files_provider = messages.provider("files")
1180
+ self.assertIsNotNone(files_provider)
1181
+ files_provider.update(test_file)
1182
+
1183
+ # 3. 渲染并验证
1184
+ rendered = await messages.render_latest()
1185
+ self.assertEqual(len(rendered), 1)
1186
+ content = rendered[0]['content']
1187
+
1188
+ # 4. 验证不可见的 provider 的内容没有出现
1189
+ self.assertNotIn("<tools>", content)
1190
+ self.assertNotIn("should_not_appear", content)
1191
+
1192
+ # 5. 验证可见的 provider 的内容正常出现
1193
+ self.assertIn("<latest_file_content>", content)
1194
+ self.assertIn("visible content", content)
1195
+
1196
+ finally:
1197
+ if os.path.exists(test_file):
1198
+ os.remove(test_file)
1199
+
1200
+ async def test_zc_message_provider_by_name(self):
1162
1201
  """测试是否可以通过名称从 Message 对象中获取 provider"""
1163
1202
  # 1. 创建一个包含命名 provider 的 Message
1164
1203
  message = UserMessage(
@@ -1182,7 +1221,7 @@ Current time: {Texts(lambda: datetime.now().strftime("%Y-%m-%d %H:%M:%S"))}
1182
1221
  non_existent_provider = message.provider("non_existent")
1183
1222
  self.assertIsNone(non_existent_provider)
1184
1223
 
1185
- async def test_zc_slicing_support(self):
1224
+ async def test_zd_slicing_support(self):
1186
1225
  """测试 Messages 对象是否支持切片操作"""
1187
1226
  m1 = SystemMessage("1")
1188
1227
  m2 = UserMessage("2")
@@ -1217,7 +1256,7 @@ Current time: {Texts(lambda: datetime.now().strftime("%Y-%m-%d %H:%M:%S"))}
1217
1256
  self.assertEqual(len(sliced_single), 1)
1218
1257
  self.assertIs(sliced_single[0], m3)
1219
1258
 
1220
- async def test_zd_slice_assignment(self):
1259
+ async def test_ze_slice_assignment(self):
1221
1260
  """测试 Messages 对象的切片赋值功能"""
1222
1261
  # 1. Setup initial Messages objects
1223
1262
  m1 = SystemMessage("1")
@@ -1258,7 +1297,7 @@ Current time: {Texts(lambda: datetime.now().strftime("%Y-%m-%d %H:%M:%S"))}
1258
1297
  self.assertEqual(messages3[2].content, "C")
1259
1298
  self.assertIsInstance(messages3[1], AssistantMessage)
1260
1299
 
1261
- async def test_ze_fstring_lambda_serialization(self):
1300
+ async def test_zf_fstring_lambda_serialization(self):
1262
1301
  """测试包含 lambda 的 f-string 消息是否可以被序列化和反序列化"""
1263
1302
  import platform
1264
1303
  import os
@@ -1302,6 +1341,51 @@ Current time: {Texts(lambda: datetime.now().strftime("%Y-%m-%d %H:%M:%S"))}
1302
1341
  if os.path.exists(test_file_path):
1303
1342
  os.remove(test_file_path)
1304
1343
 
1344
+ async def test_zg_provider_plus_message_addition(self):
1345
+ """测试所有 ContextProvider 子类与 Message 子类相加的功能"""
1346
+ # 1. 准备 providers
1347
+ text_provider = Texts("Some text.")
1348
+ tools_provider = Tools([{"name": "a_tool"}])
1349
+
1350
+ test_file = "test_provider_addition.txt"
1351
+ with open(test_file, "w") as f: f.write("File content.")
1352
+ files_provider = Files(test_file)
1353
+
1354
+ providers_to_test = [text_provider, tools_provider, files_provider]
1355
+
1356
+ # 2. 准备 message aclsdd
1357
+ messages_to_test = [
1358
+ UserMessage(Texts("Initial user message.")),
1359
+ SystemMessage(Texts("Initial system message.")),
1360
+ AssistantMessage(Texts("Initial assistant message."))
1361
+ ]
1362
+
1363
+ try:
1364
+ for provider in providers_to_test:
1365
+ for message in messages_to_test:
1366
+ with self.subTest(provider=type(provider).__name__, message=type(message).__name__):
1367
+ # 执行加法操作
1368
+ result_message = provider + message
1369
+
1370
+ # 验证结果类型是否与原始 message 相同
1371
+ self.assertIsInstance(result_message, type(message), f"结果应为 {type(message).__name__} 类型")
1372
+
1373
+ # 验证 provider 数量
1374
+ self.assertEqual(len(result_message), 2, "结果消息应包含两个 provider")
1375
+
1376
+ # 验证 provider 的类型和顺序
1377
+ result_providers = result_message.provider()
1378
+ self.assertIsInstance(result_providers[0], type(provider), f"第一个 provider 应为 {type(provider).__name__} 类型")
1379
+ self.assertIsInstance(result_providers[1], Texts, "第二个 provider 应为 Texts 类型")
1380
+
1381
+ # 验证原始消息没有被修改
1382
+ self.assertEqual(len(message), 1)
1383
+
1384
+ finally:
1385
+ # 3. 清理文件
1386
+ if os.path.exists(test_file):
1387
+ os.remove(test_file)
1388
+
1305
1389
 
1306
1390
  # ==============================================================================
1307
1391
  # 6. 演示
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: aient
3
- Version: 1.2.14
3
+ Version: 1.2.16
4
4
  Summary: Aient: The Awakening of Agent.
5
5
  Requires-Python: >=3.11
6
6
  Description-Content-Type: text/markdown
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "aient"
3
- version = "1.2.14"
3
+ version = "1.2.16"
4
4
  description = "Aient: The Awakening of Agent."
5
5
  readme = "README.md"
6
6
  requires-python = ">=3.11"
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes