sophhub 0.2.0 → 0.2.2
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.
- package/package.json +1 -1
- package/skills/compact-context/skill.json +20 -0
- package/skills/compact-context/src/SKILL.md +133 -0
- package/skills/compact-context/src/scripts/check.sh +381 -0
- package/skills/compact-context/src/scripts/set-keep-recent.mjs +1337 -0
- package/skills/compact-context/src/scripts/setup.sh +96 -0
- package/skills/feishu-notes-assistant-universal/skill.json +20 -0
- package/skills/feishu-notes-assistant-universal/src/README.md +55 -0
- package/skills/feishu-notes-assistant-universal/src/SKILL.md +159 -0
- package/skills/feishu-notes-assistant-universal/src/bin/linux-amd64/lark-cli-openclaw +0 -0
- package/skills/feishu-notes-assistant-universal/src/bin/linux-arm64/lark-cli-openclaw +0 -0
- package/skills/feishu-notes-assistant-universal/src/scripts/_resolve_lark_cli.py +58 -0
- package/skills/feishu-notes-assistant-universal/src/scripts/openclaw_meeting_minutes.py +462 -0
- package/skills/feishu-notes-assistant-universal/src/scripts/openclaw_notes_crud.py +547 -0
- package/skills/feishu-notes-assistant-universal/src/scripts/openclaw_notes_crud_test.py +181 -0
- package/skills/feishu-notes-assistant-universal/src/scripts/run_meeting_minutes.py +80 -0
- package/skills/feishu-notes-assistant-universal/src/scripts/run_meeting_minutes.sh +5 -0
- package/skills/feishu-notes-assistant-universal/src/scripts/run_note_crud.py +32 -0
- package/skills/feishu-notes-assistant-universal/src/scripts/run_note_crud.sh +5 -0
- package/skills/image-classify/skill.json +5 -5
- package/skills/image-classify/src/SKILL.md +60 -67
- package/skills/image-classify/src/scripts/face_search.py +400 -15
- package/skills/image-classify/src/scripts/send_dm_message.py +332 -0
- package/skills/md2pdf-converter/skill.json +20 -0
- package/skills/md2pdf-converter/src/SKILL.md +244 -0
- package/skills/md2pdf-converter/src/_meta.json +6 -0
- package/skills/md2pdf-converter/src/scripts/generate_emoji_mapping.py +74 -0
- package/skills/md2pdf-converter/src/scripts/md2pdf-local.sh +291 -0
- package/skills/sophnet-bot-client/skill.json +20 -0
- package/skills/sophnet-bot-client/src/SKILL.md +255 -0
- package/skills/sophnet-bot-client/src/pyproject.toml +13 -0
- package/skills/sophnet-bot-client/src/scripts/__init__.py +0 -0
- package/skills/sophnet-bot-client/src/scripts/bot_client_proxy.py +165 -0
- package/skills/sophnet-bot-client/src/scripts/bot_client_safe.sh +29 -0
- package/skills/sophnet-bot-client/src/scripts/bot_client_setup.py +502 -0
- package/skills/sophnet-bot-client/src/tests/__init__.py +0 -0
- package/skills/sophnet-bot-client/src/tests/test_bot_client_proxy.py +255 -0
- package/skills/sophnet-bot-client/src/tests/test_bot_client_setup.py +679 -0
- package/skills/sophnet-bot-client/src/uv.lock +8 -0
- package/skills/sophnet-docx/skill.json +20 -0
- package/skills/sophnet-docx/src/SKILL.md +463 -0
- package/skills/sophnet-docx/src/package-lock.json +208 -0
- package/skills/sophnet-docx/src/package.json +16 -0
- package/skills/sophnet-docx/src/pyproject.toml +11 -0
- package/skills/sophnet-docx/src/scripts/__init__.py +1 -0
- package/skills/sophnet-docx/src/scripts/accept_changes.py +135 -0
- package/skills/sophnet-docx/src/scripts/comment.py +318 -0
- package/skills/sophnet-docx/src/scripts/ensure_uv_env.sh +68 -0
- package/skills/sophnet-docx/src/scripts/office/helpers/__init__.py +0 -0
- package/skills/sophnet-docx/src/scripts/office/helpers/merge_runs.py +199 -0
- package/skills/sophnet-docx/src/scripts/office/helpers/simplify_redlines.py +197 -0
- package/skills/sophnet-docx/src/scripts/office/pack.py +159 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/mce/mc.xsd +75 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/microsoft/wml-2010.xsd +560 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/microsoft/wml-2012.xsd +67 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/microsoft/wml-2018.xsd +14 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/microsoft/wml-cex-2018.xsd +20 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/microsoft/wml-cid-2016.xsd +13 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
- package/skills/sophnet-docx/src/scripts/office/schemas/microsoft/wml-symex-2015.xsd +8 -0
- package/skills/sophnet-docx/src/scripts/office/soffice.py +183 -0
- package/skills/sophnet-docx/src/scripts/office/unpack.py +132 -0
- package/skills/sophnet-docx/src/scripts/office/validate.py +111 -0
- package/skills/sophnet-docx/src/scripts/office/validators/__init__.py +15 -0
- package/skills/sophnet-docx/src/scripts/office/validators/base.py +847 -0
- package/skills/sophnet-docx/src/scripts/office/validators/docx.py +446 -0
- package/skills/sophnet-docx/src/scripts/office/validators/pptx.py +275 -0
- package/skills/sophnet-docx/src/scripts/office/validators/redlining.py +247 -0
- package/skills/sophnet-docx/src/scripts/templates/comments.xml +3 -0
- package/skills/sophnet-docx/src/scripts/templates/commentsExtended.xml +3 -0
- package/skills/sophnet-docx/src/scripts/templates/commentsExtensible.xml +3 -0
- package/skills/sophnet-docx/src/scripts/templates/commentsIds.xml +3 -0
- package/skills/sophnet-docx/src/scripts/templates/people.xml +3 -0
- package/skills/sophnet-docx/src/scripts/upload_file.sh +96 -0
- package/skills/sophnet-docx/src/uv.lock +320 -0
- package/skills/sophnet-pdf/skill.json +20 -0
- package/skills/sophnet-pdf/src/SKILL.md +413 -0
- package/skills/sophnet-pdf/src/forms.md +297 -0
- package/skills/sophnet-pdf/src/pyproject.toml +14 -0
- package/skills/sophnet-pdf/src/reference.md +612 -0
- package/skills/sophnet-pdf/src/scripts/check_bounding_boxes.py +65 -0
- package/skills/sophnet-pdf/src/scripts/check_fillable_fields.py +11 -0
- package/skills/sophnet-pdf/src/scripts/convert_pdf_to_images.py +33 -0
- package/skills/sophnet-pdf/src/scripts/create_validation_image.py +37 -0
- package/skills/sophnet-pdf/src/scripts/enhance_tutorial.py +558 -0
- package/skills/sophnet-pdf/src/scripts/ensure_uv_env.sh +68 -0
- package/skills/sophnet-pdf/src/scripts/extract_form_field_info.py +122 -0
- package/skills/sophnet-pdf/src/scripts/extract_form_structure.py +115 -0
- package/skills/sophnet-pdf/src/scripts/extract_pdf_content.py +35 -0
- package/skills/sophnet-pdf/src/scripts/fill_fillable_fields.py +98 -0
- package/skills/sophnet-pdf/src/scripts/fill_pdf_form_with_annotations.py +107 -0
- package/skills/sophnet-pdf/src/scripts/upload_file.sh +88 -0
- package/skills/sophnet-pdf/src/uv.lock +537 -0
- package/skills/sophnet-xlsx/skill.json +20 -0
- package/skills/sophnet-xlsx/src/SKILL.md +399 -0
- package/skills/sophnet-xlsx/src/pyproject.toml +11 -0
- package/skills/sophnet-xlsx/src/scripts/ensure_uv_env.sh +68 -0
- package/skills/sophnet-xlsx/src/scripts/office/helpers/__init__.py +0 -0
- package/skills/sophnet-xlsx/src/scripts/office/helpers/merge_runs.py +199 -0
- package/skills/sophnet-xlsx/src/scripts/office/helpers/simplify_redlines.py +197 -0
- package/skills/sophnet-xlsx/src/scripts/office/pack.py +159 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/mce/mc.xsd +75 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/microsoft/wml-2010.xsd +560 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/microsoft/wml-2012.xsd +67 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/microsoft/wml-2018.xsd +14 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/microsoft/wml-cex-2018.xsd +20 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/microsoft/wml-cid-2016.xsd +13 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
- package/skills/sophnet-xlsx/src/scripts/office/schemas/microsoft/wml-symex-2015.xsd +8 -0
- package/skills/sophnet-xlsx/src/scripts/office/soffice.py +183 -0
- package/skills/sophnet-xlsx/src/scripts/office/unpack.py +132 -0
- package/skills/sophnet-xlsx/src/scripts/office/validate.py +111 -0
- package/skills/sophnet-xlsx/src/scripts/office/validators/__init__.py +15 -0
- package/skills/sophnet-xlsx/src/scripts/office/validators/base.py +847 -0
- package/skills/sophnet-xlsx/src/scripts/office/validators/docx.py +446 -0
- package/skills/sophnet-xlsx/src/scripts/office/validators/pptx.py +275 -0
- package/skills/sophnet-xlsx/src/scripts/office/validators/redlining.py +247 -0
- package/skills/sophnet-xlsx/src/scripts/recalc.py +184 -0
- package/skills/sophnet-xlsx/src/scripts/upload_file.sh +96 -0
- package/skills/sophnet-xlsx/src/uv.lock +319 -0
- package/skills/wechat-article-publisher/skill.json +20 -0
- package/skills/wechat-article-publisher/src/SKILL.md +60 -0
- package/skills/wechat-article-publisher/src/config.json +7 -0
- package/skills/wechat-article-publisher/src/pyproject.toml +12 -0
- package/skills/wechat-article-publisher/src/scripts/publish_wechat.py +825 -0
|
@@ -0,0 +1,255 @@
|
|
|
1
|
+
"""Tests for bot_client_proxy.py — permission-aware Bot API proxy."""
|
|
2
|
+
|
|
3
|
+
import json
|
|
4
|
+
import os
|
|
5
|
+
import sys
|
|
6
|
+
import tempfile
|
|
7
|
+
import unittest
|
|
8
|
+
from unittest.mock import patch
|
|
9
|
+
|
|
10
|
+
sys.path.insert(0, os.path.join(os.path.dirname(__file__), "..", "scripts"))
|
|
11
|
+
|
|
12
|
+
from bot_client_proxy import send_message
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
def _create_permission(admin_dir, agent_id, operations=None):
|
|
16
|
+
"""Helper: create a minimal permissions.json in admin_dir."""
|
|
17
|
+
os.makedirs(admin_dir, exist_ok=True)
|
|
18
|
+
perms = {
|
|
19
|
+
"version": 1,
|
|
20
|
+
"agents": {
|
|
21
|
+
agent_id: {
|
|
22
|
+
"agent_id": agent_id,
|
|
23
|
+
"display_name": "Test",
|
|
24
|
+
"emoji": "🔧",
|
|
25
|
+
"operations": operations or {
|
|
26
|
+
"get": True, "search": True,
|
|
27
|
+
"add": False, "update": False, "delete": False,
|
|
28
|
+
"visit": False, "visits": False, "stats": False,
|
|
29
|
+
"remind": False, "export": False, "import": False,
|
|
30
|
+
},
|
|
31
|
+
"scope": {"levels": [], "groups": [], "tags": []},
|
|
32
|
+
}
|
|
33
|
+
},
|
|
34
|
+
}
|
|
35
|
+
with open(os.path.join(admin_dir, "permissions.json"), "w") as f:
|
|
36
|
+
json.dump(perms, f)
|
|
37
|
+
|
|
38
|
+
|
|
39
|
+
def _create_credentials(admin_dir, agent_id, url="http://localhost:8080/bot-api/chat"):
|
|
40
|
+
"""Helper: create bot_credentials.json in admin_dir/<agent_id>/."""
|
|
41
|
+
creds_dir = os.path.join(admin_dir, agent_id)
|
|
42
|
+
os.makedirs(creds_dir, exist_ok=True)
|
|
43
|
+
creds = {
|
|
44
|
+
"url": url,
|
|
45
|
+
"secret": "test-secret",
|
|
46
|
+
"remote_agent_id": "intern-helper",
|
|
47
|
+
}
|
|
48
|
+
with open(os.path.join(creds_dir, "bot_credentials.json"), "w") as f:
|
|
49
|
+
json.dump(creds, f)
|
|
50
|
+
|
|
51
|
+
|
|
52
|
+
class TestSendMessagePermissionCheck(unittest.TestCase):
|
|
53
|
+
"""Test that send_message checks permissions before sending."""
|
|
54
|
+
|
|
55
|
+
def setUp(self):
|
|
56
|
+
self.tmpdir = tempfile.mkdtemp()
|
|
57
|
+
self.admin_dir = os.path.join(self.tmpdir, "admin")
|
|
58
|
+
self.agent_id = "test-agent"
|
|
59
|
+
|
|
60
|
+
def tearDown(self):
|
|
61
|
+
import shutil
|
|
62
|
+
shutil.rmtree(self.tmpdir, ignore_errors=True)
|
|
63
|
+
|
|
64
|
+
def test_rejects_unknown_agent(self):
|
|
65
|
+
"""Agent without permission record should be rejected."""
|
|
66
|
+
os.makedirs(self.admin_dir, exist_ok=True)
|
|
67
|
+
with open(os.path.join(self.admin_dir, "permissions.json"), "w") as f:
|
|
68
|
+
json.dump({"version": 1, "agents": {}}, f)
|
|
69
|
+
|
|
70
|
+
result = send_message(
|
|
71
|
+
agent_id=self.agent_id,
|
|
72
|
+
message="hello",
|
|
73
|
+
admin_dir=self.admin_dir,
|
|
74
|
+
)
|
|
75
|
+
self.assertEqual(result["status"], "error")
|
|
76
|
+
self.assertIn("权限记录", result["message"])
|
|
77
|
+
|
|
78
|
+
def test_rejects_missing_credentials(self):
|
|
79
|
+
"""Agent with permissions but no credentials should be rejected."""
|
|
80
|
+
_create_permission(self.admin_dir, self.agent_id)
|
|
81
|
+
|
|
82
|
+
result = send_message(
|
|
83
|
+
agent_id=self.agent_id,
|
|
84
|
+
message="hello",
|
|
85
|
+
admin_dir=self.admin_dir,
|
|
86
|
+
)
|
|
87
|
+
self.assertEqual(result["status"], "error")
|
|
88
|
+
self.assertIn("凭据", result["message"])
|
|
89
|
+
|
|
90
|
+
@patch("bot_client_proxy.subprocess.run")
|
|
91
|
+
def test_sends_with_permissions(self, mock_run):
|
|
92
|
+
"""Valid agent should send request with senderPermissions attached."""
|
|
93
|
+
_create_permission(self.admin_dir, self.agent_id, {
|
|
94
|
+
"get": True, "search": True, "delete": False,
|
|
95
|
+
"add": False, "update": False,
|
|
96
|
+
"visit": False, "visits": False, "stats": False,
|
|
97
|
+
"remind": False, "export": False, "import": False,
|
|
98
|
+
})
|
|
99
|
+
_create_credentials(self.admin_dir, self.agent_id)
|
|
100
|
+
|
|
101
|
+
mock_run.return_value = type("R", (), {
|
|
102
|
+
"returncode": 0,
|
|
103
|
+
"stdout": json.dumps({"reply": "ok", "sessionKey": "sess-1"}),
|
|
104
|
+
"stderr": "",
|
|
105
|
+
})()
|
|
106
|
+
|
|
107
|
+
result = send_message(
|
|
108
|
+
agent_id=self.agent_id,
|
|
109
|
+
message="查询客户",
|
|
110
|
+
admin_dir=self.admin_dir,
|
|
111
|
+
)
|
|
112
|
+
self.assertEqual(result["status"], "ok")
|
|
113
|
+
self.assertEqual(result["reply"], "ok")
|
|
114
|
+
|
|
115
|
+
call_args = mock_run.call_args[0][0]
|
|
116
|
+
body_json = call_args[-1]
|
|
117
|
+
body = json.loads(body_json)
|
|
118
|
+
self.assertIn("senderPermissions", body)
|
|
119
|
+
self.assertTrue(body["senderPermissions"]["operations"]["get"])
|
|
120
|
+
self.assertFalse(body["senderPermissions"]["operations"]["delete"])
|
|
121
|
+
self.assertEqual(body["senderId"], self.agent_id)
|
|
122
|
+
self.assertEqual(body["agentId"], "intern-helper")
|
|
123
|
+
|
|
124
|
+
@patch("bot_client_proxy.subprocess.run")
|
|
125
|
+
def test_stable_chat_id(self, mock_run):
|
|
126
|
+
"""chatId should always be agent_id to keep a stable session."""
|
|
127
|
+
_create_permission(self.admin_dir, self.agent_id)
|
|
128
|
+
_create_credentials(self.admin_dir, self.agent_id)
|
|
129
|
+
|
|
130
|
+
mock_run.return_value = type("R", (), {
|
|
131
|
+
"returncode": 0,
|
|
132
|
+
"stdout": json.dumps({"reply": "first", "sessionKey": "agent:intern-helper:bot-api:test-agent-api:dm:test-agent"}),
|
|
133
|
+
"stderr": "",
|
|
134
|
+
})()
|
|
135
|
+
|
|
136
|
+
send_message(
|
|
137
|
+
agent_id=self.agent_id,
|
|
138
|
+
message="first message",
|
|
139
|
+
admin_dir=self.admin_dir,
|
|
140
|
+
)
|
|
141
|
+
|
|
142
|
+
call_args = mock_run.call_args[0][0]
|
|
143
|
+
body = json.loads(call_args[-1])
|
|
144
|
+
self.assertEqual(body["chatId"], self.agent_id)
|
|
145
|
+
|
|
146
|
+
mock_run.return_value = type("R", (), {
|
|
147
|
+
"returncode": 0,
|
|
148
|
+
"stdout": json.dumps({"reply": "second", "sessionKey": "agent:intern-helper:bot-api:test-agent-api:dm:test-agent"}),
|
|
149
|
+
"stderr": "",
|
|
150
|
+
})()
|
|
151
|
+
|
|
152
|
+
send_message(
|
|
153
|
+
agent_id=self.agent_id,
|
|
154
|
+
message="second message",
|
|
155
|
+
admin_dir=self.admin_dir,
|
|
156
|
+
)
|
|
157
|
+
|
|
158
|
+
call_args = mock_run.call_args[0][0]
|
|
159
|
+
body = json.loads(call_args[-1])
|
|
160
|
+
self.assertEqual(body["chatId"], self.agent_id)
|
|
161
|
+
|
|
162
|
+
@patch("bot_client_proxy.subprocess.run")
|
|
163
|
+
def test_handles_curl_failure(self, mock_run):
|
|
164
|
+
_create_permission(self.admin_dir, self.agent_id)
|
|
165
|
+
_create_credentials(self.admin_dir, self.agent_id)
|
|
166
|
+
|
|
167
|
+
mock_run.return_value = type("R", (), {
|
|
168
|
+
"returncode": 7,
|
|
169
|
+
"stdout": "",
|
|
170
|
+
"stderr": "Connection refused",
|
|
171
|
+
})()
|
|
172
|
+
|
|
173
|
+
result = send_message(
|
|
174
|
+
agent_id=self.agent_id,
|
|
175
|
+
message="hello",
|
|
176
|
+
admin_dir=self.admin_dir,
|
|
177
|
+
)
|
|
178
|
+
self.assertEqual(result["status"], "error")
|
|
179
|
+
self.assertIn("curl", result["message"])
|
|
180
|
+
|
|
181
|
+
@patch("bot_client_proxy.subprocess.run")
|
|
182
|
+
def test_handles_invalid_json_response(self, mock_run):
|
|
183
|
+
_create_permission(self.admin_dir, self.agent_id)
|
|
184
|
+
_create_credentials(self.admin_dir, self.agent_id)
|
|
185
|
+
|
|
186
|
+
mock_run.return_value = type("R", (), {
|
|
187
|
+
"returncode": 0,
|
|
188
|
+
"stdout": "not json",
|
|
189
|
+
"stderr": "",
|
|
190
|
+
})()
|
|
191
|
+
|
|
192
|
+
result = send_message(
|
|
193
|
+
agent_id=self.agent_id,
|
|
194
|
+
message="hello",
|
|
195
|
+
admin_dir=self.admin_dir,
|
|
196
|
+
)
|
|
197
|
+
self.assertEqual(result["status"], "error")
|
|
198
|
+
self.assertIn("无法解析", result["message"])
|
|
199
|
+
|
|
200
|
+
@patch("bot_client_proxy.subprocess.run")
|
|
201
|
+
def test_handles_timeout(self, mock_run):
|
|
202
|
+
import subprocess
|
|
203
|
+
_create_permission(self.admin_dir, self.agent_id)
|
|
204
|
+
_create_credentials(self.admin_dir, self.agent_id)
|
|
205
|
+
|
|
206
|
+
mock_run.side_effect = subprocess.TimeoutExpired(cmd="curl", timeout=120)
|
|
207
|
+
|
|
208
|
+
result = send_message(
|
|
209
|
+
agent_id=self.agent_id,
|
|
210
|
+
message="hello",
|
|
211
|
+
admin_dir=self.admin_dir,
|
|
212
|
+
)
|
|
213
|
+
self.assertEqual(result["status"], "error")
|
|
214
|
+
self.assertIn("超时", result["message"])
|
|
215
|
+
|
|
216
|
+
@patch("bot_client_proxy.subprocess.run")
|
|
217
|
+
def test_scope_included_in_request(self, mock_run):
|
|
218
|
+
"""Sender scope restrictions should be forwarded."""
|
|
219
|
+
os.makedirs(self.admin_dir, exist_ok=True)
|
|
220
|
+
perms = {
|
|
221
|
+
"version": 1,
|
|
222
|
+
"agents": {
|
|
223
|
+
self.agent_id: {
|
|
224
|
+
"agent_id": self.agent_id,
|
|
225
|
+
"display_name": "Scoped",
|
|
226
|
+
"emoji": "🔧",
|
|
227
|
+
"operations": {"get": True, "search": True},
|
|
228
|
+
"scope": {"levels": ["normal", "new"], "groups": [], "tags": []},
|
|
229
|
+
}
|
|
230
|
+
},
|
|
231
|
+
}
|
|
232
|
+
with open(os.path.join(self.admin_dir, "permissions.json"), "w") as f:
|
|
233
|
+
json.dump(perms, f)
|
|
234
|
+
_create_credentials(self.admin_dir, self.agent_id)
|
|
235
|
+
|
|
236
|
+
mock_run.return_value = type("R", (), {
|
|
237
|
+
"returncode": 0,
|
|
238
|
+
"stdout": json.dumps({"reply": "ok", "sessionKey": ""}),
|
|
239
|
+
"stderr": "",
|
|
240
|
+
})()
|
|
241
|
+
|
|
242
|
+
send_message(
|
|
243
|
+
agent_id=self.agent_id,
|
|
244
|
+
message="查询",
|
|
245
|
+
admin_dir=self.admin_dir,
|
|
246
|
+
)
|
|
247
|
+
|
|
248
|
+
body = json.loads(mock_run.call_args[0][0][-1])
|
|
249
|
+
self.assertEqual(
|
|
250
|
+
body["senderPermissions"]["scope"]["levels"], ["normal", "new"]
|
|
251
|
+
)
|
|
252
|
+
|
|
253
|
+
|
|
254
|
+
if __name__ == "__main__":
|
|
255
|
+
unittest.main()
|