aient 1.2.14__py3-none-any.whl → 1.2.16__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.
- aient/architext/architext/core.py +17 -10
- aient/architext/test/test.py +88 -4
- {aient-1.2.14.dist-info → aient-1.2.16.dist-info}/METADATA +1 -1
- {aient-1.2.14.dist-info → aient-1.2.16.dist-info}/RECORD +7 -7
- {aient-1.2.14.dist-info → aient-1.2.16.dist-info}/WHEEL +0 -0
- {aient-1.2.14.dist-info → aient-1.2.16.dist-info}/licenses/LICENSE +0 -0
- {aient-1.2.14.dist-info → aient-1.2.16.dist-info}/top_level.txt +0 -0
@@ -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 =
|
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
|
aient/architext/test/test.py
CHANGED
@@ -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
|
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
|
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
|
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
|
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,8 +1,8 @@
|
|
1
1
|
aient/__init__.py,sha256=SRfF7oDVlOOAi6nGKiJIUK6B_arqYLO9iSMp-2IZZps,21
|
2
2
|
aient/architext/architext/__init__.py,sha256=79Ih1151rfcqZdr7F8HSZSTs_iT2SKd1xCkehMsXeXs,19
|
3
|
-
aient/architext/architext/core.py,sha256=
|
3
|
+
aient/architext/architext/core.py,sha256=KLRixiSd6q2B5Yq62U0q-ebPMySbzJNc8g8VQTsf1C0,28894
|
4
4
|
aient/architext/test/openai_client.py,sha256=Dqtbmubv6vwF8uBqcayG0kbsiO65of7sgU2-DRBi-UM,4590
|
5
|
-
aient/architext/test/test.py,sha256=
|
5
|
+
aient/architext/test/test.py,sha256=KKQwaxsumYQkbUl2M1tIn0gJyymGH9P8tWxd7A8HRDc,64739
|
6
6
|
aient/architext/test/test_save_load.py,sha256=o8DqH6gDYZkFkQy-a7blqLtJTRj5e4a-Lil48pJ0V3g,3260
|
7
7
|
aient/core/__init__.py,sha256=NxjebTlku35S4Dzr16rdSqSTWUvvwEeACe8KvHJnjPg,34
|
8
8
|
aient/core/log_config.py,sha256=kz2_yJv1p-o3lUQOwA3qh-LSc3wMHv13iCQclw44W9c,274
|
@@ -35,8 +35,8 @@ aient/plugins/write_file.py,sha256=Jt8fOEwqhYiSWpCbwfAr1xoi_BmFnx3076GMhuL06uI,3
|
|
35
35
|
aient/utils/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
36
36
|
aient/utils/prompt.py,sha256=UcSzKkFE4-h_1b6NofI6xgk3GoleqALRKY8VBaXLjmI,11311
|
37
37
|
aient/utils/scripts.py,sha256=VqtK4RFEx7KxkmcqG3lFDS1DxoNlFFGErEjopVcc8IE,40974
|
38
|
-
aient-1.2.
|
39
|
-
aient-1.2.
|
40
|
-
aient-1.2.
|
41
|
-
aient-1.2.
|
42
|
-
aient-1.2.
|
38
|
+
aient-1.2.16.dist-info/licenses/LICENSE,sha256=XNdbcWldt0yaNXXWB_Bakoqnxb3OVhUft4MgMA_71ds,1051
|
39
|
+
aient-1.2.16.dist-info/METADATA,sha256=nPvLpHDPPi-Z8jNKlgdM8m7GNvfhohV5AAvbiHIzHS0,4842
|
40
|
+
aient-1.2.16.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
|
41
|
+
aient-1.2.16.dist-info/top_level.txt,sha256=3oXzrP5sAVvyyqabpeq8A2_vfMtY554r4bVE-OHBrZk,6
|
42
|
+
aient-1.2.16.dist-info/RECORD,,
|
File without changes
|
File without changes
|
File without changes
|