sophhub 0.1.0

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 (125) hide show
  1. package/bin/sophhub.js +21 -0
  2. package/package.json +32 -0
  3. package/skills/VERSIONS.md +27 -0
  4. package/skills/builtin/clawhub/SKILL.md +77 -0
  5. package/skills/builtin/flight-booking/SKILL.md +288 -0
  6. package/skills/builtin/flight-booking/scripts/flight_booking.py +1232 -0
  7. package/skills/builtin/inventory-management/SKILL.md +241 -0
  8. package/skills/builtin/inventory-management/scripts/inventory.py +1844 -0
  9. package/skills/builtin/schedule-reminder/SKILL.md +619 -0
  10. package/skills/builtin/schedule-reminder/schedule_template.md +68 -0
  11. package/skills/builtin/schedule-reminder/scripts/append_event.py +204 -0
  12. package/skills/builtin/schedule-reminder/scripts/create_reminders.sh +163 -0
  13. package/skills/builtin/schedule-reminder/scripts/daily_activate.sh +175 -0
  14. package/skills/builtin/schedule-reminder/scripts/parse_schedule.py +704 -0
  15. package/skills/builtin/schedule-reminder/scripts/setup.sh +242 -0
  16. package/skills/builtin/schedule-reminder//347/224/250/346/210/267/346/214/207/345/215/227.md +311 -0
  17. package/skills/builtin/skill-creator/SKILL.md +370 -0
  18. package/skills/builtin/skill-creator/license.txt +202 -0
  19. package/skills/builtin/skill-creator/scripts/init_skill.py +378 -0
  20. package/skills/builtin/skill-creator/scripts/package_skill.py +111 -0
  21. package/skills/builtin/skill-creator/scripts/quick_validate.py +101 -0
  22. package/skills/builtin/sophnet-customer-management/SKILL.md +271 -0
  23. package/skills/builtin/sophnet-customer-management/pyproject.toml +15 -0
  24. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_cli/__init__.py +2 -0
  25. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_cli/__main__.py +5 -0
  26. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_cli/cli.py +67 -0
  27. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_cli/commands/__init__.py +2 -0
  28. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_cli/commands/customer.py +60 -0
  29. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_cli/commands/export_file.py +18 -0
  30. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_cli/commands/import_file.py +15 -0
  31. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_cli/commands/reminder.py +26 -0
  32. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_cli/commands/schema.py +28 -0
  33. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_cli/config.py +54 -0
  34. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_core/__init__.py +2 -0
  35. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_core/exporter.py +85 -0
  36. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_core/models.py +84 -0
  37. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_core/normalizer.py +144 -0
  38. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_core/parser.py +241 -0
  39. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_core/query.py +109 -0
  40. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_core/reminder.py +121 -0
  41. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_core/repository.py +397 -0
  42. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_core/schema.py +106 -0
  43. package/skills/builtin/sophnet-customer-management/src/customer_mgmt_core/service.py +565 -0
  44. package/skills/builtin/sophnet-customer-management/uv.lock +48 -0
  45. package/skills/builtin/sophnet-customized-marketing/SKILL.md +144 -0
  46. package/skills/builtin/sophnet-customized-marketing/playbooks/campaign-planning.md +187 -0
  47. package/skills/builtin/sophnet-customized-marketing/playbooks/content-generation.md +124 -0
  48. package/skills/builtin/sophnet-customized-marketing/playbooks/marketing-calendar.md +59 -0
  49. package/skills/builtin/sophnet-customized-marketing/playbooks/multi-channel-bundle.md +94 -0
  50. package/skills/builtin/sophnet-customized-marketing/playbooks/poster-generation.md +182 -0
  51. package/skills/builtin/sophnet-customized-marketing/playbooks/style-profile-workflow.md +103 -0
  52. package/skills/builtin/sophnet-customized-marketing/pyproject.toml +9 -0
  53. package/skills/builtin/sophnet-customized-marketing/references/campaign-mechanics.md +168 -0
  54. package/skills/builtin/sophnet-customized-marketing/references/content-safety.md +26 -0
  55. package/skills/builtin/sophnet-customized-marketing/references/marketing-date-checklist.md +99 -0
  56. package/skills/builtin/sophnet-customized-marketing/references/platform-writing-guidelines.md +88 -0
  57. package/skills/builtin/sophnet-customized-marketing/references/quality-checklist.md +44 -0
  58. package/skills/builtin/sophnet-customized-marketing/scripts/generate_poster.py +585 -0
  59. package/skills/builtin/sophnet-customized-marketing/scripts/style_profile.py +215 -0
  60. package/skills/builtin/sophnet-face-search/SKILL.md +115 -0
  61. package/skills/builtin/sophnet-face-search/pyproject.toml +11 -0
  62. package/skills/builtin/sophnet-face-search/scripts/face_search.py +336 -0
  63. package/skills/builtin/sophnet-face-search/uv.lock +508 -0
  64. package/skills/builtin/sophnet-image-edit/SKILL.md +140 -0
  65. package/skills/builtin/sophnet-image-edit/pyproject.toml +9 -0
  66. package/skills/builtin/sophnet-image-edit/scripts/edit_and_preview.sh +68 -0
  67. package/skills/builtin/sophnet-image-edit/scripts/edit_image.py +279 -0
  68. package/skills/builtin/sophnet-image-edit/uv.lock +234 -0
  69. package/skills/builtin/sophnet-image-generate/SKILL.md +62 -0
  70. package/skills/builtin/sophnet-image-generate/pyproject.toml +9 -0
  71. package/skills/builtin/sophnet-image-generate/scripts/generate_image.py +156 -0
  72. package/skills/builtin/sophnet-image-generate/uv.lock +234 -0
  73. package/skills/builtin/sophnet-image-ocr/SKILL.md +167 -0
  74. package/skills/builtin/sophnet-image-ocr/pyproject.toml +13 -0
  75. package/skills/builtin/sophnet-image-ocr/scripts/ocr.py +226 -0
  76. package/skills/builtin/sophnet-image-ocr/uv.lock +234 -0
  77. package/skills/builtin/sophnet-infinite-talk/SKILL.md +140 -0
  78. package/skills/builtin/sophnet-infinite-talk/pyproject.toml +9 -0
  79. package/skills/builtin/sophnet-infinite-talk/scripts/gen.py +172 -0
  80. package/skills/builtin/sophnet-oss/SKILL.md +109 -0
  81. package/skills/builtin/sophnet-oss/pyproject.toml +8 -0
  82. package/skills/builtin/sophnet-oss/scripts/upload_file.py +43 -0
  83. package/skills/builtin/sophnet-qa-install/SKILL.md +210 -0
  84. package/skills/builtin/sophnet-qa-install/pyproject.toml +6 -0
  85. package/skills/builtin/sophnet-qa-install/scripts/backup_md.py +35 -0
  86. package/skills/builtin/sophnet-qa-install/scripts/check_installed.py +143 -0
  87. package/skills/builtin/sophnet-qa-install/scripts/update_config.py +142 -0
  88. package/skills/builtin/sophnet-qa-install/scripts/update_md.py +73 -0
  89. package/skills/builtin/sophnet-training-install/SKILL.md +211 -0
  90. package/skills/builtin/sophnet-training-install/pyproject.toml +6 -0
  91. package/skills/builtin/sophnet-training-install/scripts/backup_md.py +35 -0
  92. package/skills/builtin/sophnet-training-install/scripts/check_installed.py +144 -0
  93. package/skills/builtin/sophnet-training-install/scripts/update_config.py +142 -0
  94. package/skills/builtin/sophnet-training-install/scripts/update_md.py +73 -0
  95. package/skills/builtin/sophnet-tts/SKILL.md +79 -0
  96. package/skills/builtin/sophnet-tts/pyproject.toml +9 -0
  97. package/skills/builtin/sophnet-tts/scripts/gen_tts.py +130 -0
  98. package/skills/builtin/sophnet-video-generate/SKILL.md +116 -0
  99. package/skills/builtin/sophnet-video-generate/scripts/gen_video.py +304 -0
  100. package/skills/builtin/video-understand/SKILL.md +79 -0
  101. package/skills/builtin/video-understand/scripts/video_understand.py +204 -0
  102. package/skills/builtin/weather/SKILL.md +112 -0
  103. package/skills/builtin/web-scraper/SKILL.md +101 -0
  104. package/skills/builtin/web-scraper/scripts/scrape.py +270 -0
  105. package/skills/builtin/website-builder/SKILL.md +266 -0
  106. package/skills/builtin/website-builder/scripts/deploy_site.sh +46 -0
  107. package/skills/store/didi-ride/SKILL.md +309 -0
  108. package/skills/store/didi-ride/_meta.json +6 -0
  109. package/skills/store/didi-ride/assets/PREFERENCE.md +58 -0
  110. package/skills/store/didi-ride/package.json +15 -0
  111. package/skills/store/didi-ride/references/api_references.md +171 -0
  112. package/skills/store/didi-ride/references/error_handling.md +68 -0
  113. package/skills/store/didi-ride/references/setup.md +73 -0
  114. package/skills/store/didi-ride/references/workflow.md +150 -0
  115. package/skills/store/flyai/SKILL.md +119 -0
  116. package/skills/store/flyai/references/fliggy-fast-search.md +53 -0
  117. package/skills/store/flyai/references/search-flight.md +89 -0
  118. package/skills/store/flyai/references/search-hotels.md +57 -0
  119. package/skills/store/flyai/references/search-poi.md +49 -0
  120. package/src/commands/download.js +103 -0
  121. package/src/commands/list.js +67 -0
  122. package/src/utils/config.js +24 -0
  123. package/src/utils/gitlab.js +67 -0
  124. package/src/utils/paths.js +19 -0
  125. package/src/utils/versions.js +38 -0
@@ -0,0 +1,271 @@
1
+ ---
2
+ name: sophnet-customer-management
3
+ description: >
4
+ 客户管理助手:支持上传 Excel/CSV/Markdown 客户表创建客户库,使用自然语言创建字段、添加客户、
5
+ 查询筛选、修改删除客户,并生成生日/到期/跟进提醒。当用户要管理客户资料、导入客户名单、
6
+ 查询某月生日客户、查 30 天内到期客户、修改客户信息、导出客户数据时使用。
7
+ ---
8
+
9
+ # 客户管理助手
10
+
11
+ 这个 skill 面向“成长型小微企业主”的轻量客户管理场景。优先用对话驱动,但所有关键写操作都要经过结构化命令和必要确认。
12
+
13
+ ## 关键原则
14
+
15
+ - 先预览、后写入:导入文件时,先做预览,再让用户确认写入。
16
+ - 删除和批量修改必须确认:不要在重名或模糊命中时直接执行。
17
+ - 统一使用 Python CLI:所有确定性操作通过 `uv run --project "{baseDir}" python -m customer_mgmt_cli ...` 执行。
18
+ - 数据以 SQLite 为主:客户库主数据默认写到用户运行环境配置中的数据库路径;JSON/Markdown 只作为预览、中间结果和展示格式。
19
+
20
+ ## 首次使用初始化
21
+
22
+ - 第一次使用本 skill 时,先初始化运行时配置文件:`${HOME}/.config/sophnet-customer-management/config.env`。
23
+ - 配置文件中至少保存:
24
+ - `CUSTOMER_MGMT_DB_PATH=<sqlite绝对路径>`
25
+ - 如果用户没有指定数据库路径,则默认写为:`${HOME}/.config/sophnet-customer-management/customer_mgmt.sqlite3`
26
+ - 初始化方式:
27
+ - 第一次调用 CLI 时,如果显式传了 `--db`,CLI 会自动把该路径写入 `config.env`
28
+ - 如果没有传 `--db` 且 `config.env` 还不存在,CLI 会自动创建 `config.env` 并写入默认路径
29
+ - 后续使用本 skill 时:
30
+ - 默认先读取 `${HOME}/.config/sophnet-customer-management/config.env`
31
+ - 从 `CUSTOMER_MGMT_DB_PATH` 获取 SQLite 路径
32
+ - 因此后续命令默认不必重复传 `--db`
33
+ - 这样即使后续卸载或替换 skill 目录,默认客户数据仍然保留在用户配置目录,不会跟随 skill 一起被删除。
34
+
35
+ ### 首次初始化操作规范
36
+
37
+ - 当 agent 判断这是用户第一次使用本 skill,或当前环境中还不存在 `${HOME}/.config/sophnet-customer-management/config.env` 时,应把本轮视为“首次初始化 + 当前请求”两步一起完成,而不是单独把初始化抛给用户。
38
+ - 默认策略:
39
+ - 不额外追问数据库路径
40
+ - 直接采用默认路径 `${HOME}/.config/sophnet-customer-management/customer_mgmt.sqlite3`
41
+ - 首次 CLI 调用后自动落盘 `config.env`
42
+ - 只有在用户明确提出“数据库放到别的位置”“想自己指定路径”“多套客户库分开存”等要求时,才改为显式传 `--db <custom_path>`。
43
+ - 初始化完成后,agent 在回复中可顺带告知一次数据库位置,但不要每次都重复强调。
44
+ - 后续同一 skill 内的命令默认不再传 `--db`,除非本轮任务明确要求切换数据库。
45
+
46
+ ### 首次初始化示例对话
47
+
48
+ #### 示例 1:用户第一次直接开始用
49
+
50
+ 用户:
51
+
52
+ ```text
53
+ 帮我建一个客户表,先有姓名、手机号、生日、微信、备注。
54
+ ```
55
+
56
+ agent 内部执行思路:
57
+
58
+ - 先检查 `${HOME}/.config/sophnet-customer-management/config.env` 是否存在。
59
+ - 若不存在,则认定为首次使用。
60
+ - 不必先打断用户去确认数据库路径,直接使用默认路径 `${HOME}/.config/sophnet-customer-management/customer_mgmt.sqlite3`。
61
+ - 执行首次 CLI 调用;CLI 会自动创建:
62
+ - `${HOME}/.config/sophnet-customer-management/config.env`
63
+ - `${HOME}/.config/sophnet-customer-management/customer_mgmt.sqlite3`
64
+ - 然后继续执行本次真实任务,例如:
65
+
66
+ ```bash
67
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
68
+ schema evolve --json '{"labels":["微信"]}'
69
+ ```
70
+
71
+ agent 对用户的推荐回复风格:
72
+
73
+ ```text
74
+ 我先帮你初始化客户库,并按默认位置保存,后续都会自动复用这份客户库。
75
+ 这次会先创建基础客户表,并补一个“微信”字段。
76
+ ```
77
+
78
+ #### 示例 2:用户明确指定数据库位置
79
+
80
+ 用户:
81
+
82
+ ```text
83
+ 这个客户库不要放默认位置,放到 /data/business/customer_a.sqlite3。
84
+ ```
85
+
86
+ agent 内部执行思路:
87
+
88
+ - 识别为“用户显式指定路径”。
89
+ - 本次调用必须显式带上 `--db /data/business/customer_a.sqlite3`。
90
+ - CLI 执行后,会把该路径写入 `${HOME}/.config/sophnet-customer-management/config.env`。
91
+ - 后续若无新的指定,就默认继续使用这个路径。
92
+
93
+ 示例命令:
94
+
95
+ ```bash
96
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
97
+ --db /data/business/customer_a.sqlite3 \
98
+ schema inspect
99
+ ```
100
+
101
+ #### 示例 3:后续再次使用
102
+
103
+ 用户:
104
+
105
+ ```text
106
+ 查一下这个月过生日的客户。
107
+ ```
108
+
109
+ agent 内部执行思路:
110
+
111
+ - 默认先读取 `${HOME}/.config/sophnet-customer-management/config.env`
112
+ - 取出 `CUSTOMER_MGMT_DB_PATH`
113
+ - 直接执行查询,不再重新询问数据库路径
114
+
115
+ ```bash
116
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
117
+ customer query --json '{"birthday_month":3,"limit":20}'
118
+ ```
119
+
120
+ ## 可处理的输入
121
+
122
+ - 自然语言:
123
+ - “添加客户张三,电话 138xxxx”
124
+ - “查一下 3 月过生日的客户”
125
+ - “把张三的电话改成 139xxxx”
126
+ - “帮我建一个客户表,字段有姓名、电话、生日、备注”
127
+ - 文件:
128
+ - `.xls`
129
+ - `.xlsx`
130
+ - `.csv`
131
+ - `.md`
132
+
133
+ ## 文件导入流程
134
+
135
+ ### Step 1:定位文件
136
+
137
+ 用户上传文件后,优先在当前 agent 工作路径(working directory)下的 `media/inbound/` 中寻找最新文件;
138
+ 若用户直接给了路径,则直接使用该路径。
139
+
140
+ ### Step 2:预览导入
141
+
142
+ ```bash
143
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
144
+ import file --path "<file_path>"
145
+ ```
146
+
147
+ - 读取返回 JSON 里的 `summary`
148
+ - 向用户展示:
149
+ - 待导入客户数
150
+ - 待导入业务记录数
151
+ - 新增字段数
152
+ - 异常与告警
153
+ - 仅当用户确认后再执行真正写入
154
+
155
+ ### Step 3:确认后写入
156
+
157
+ ```bash
158
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
159
+ import file --path "<file_path>" --commit
160
+ ```
161
+
162
+ ## 自然语言创建字段
163
+
164
+ 当用户要求“建一个客户表”或“新增某个字段”时,抽取字段名称,然后执行:
165
+
166
+ ```bash
167
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
168
+ schema evolve --json '{"labels":["微信","孩子生日","最近消费"]}'
169
+ ```
170
+
171
+ 先向用户解释将新增哪些字段,再执行。
172
+
173
+ 查看当前字段:
174
+
175
+ ```bash
176
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
177
+ schema inspect
178
+ ```
179
+
180
+ ## 新增客户
181
+
182
+ ```bash
183
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
184
+ customer add --json '{
185
+ "name":"张三",
186
+ "primary_phone":"13800138000",
187
+ "birthday":"1988-03-18",
188
+ "tags":["VIP"],
189
+ "notes":"老客户",
190
+ "dynamic_fields":{"微信":"zhangsan88","喜欢项目":"补水"}
191
+ }'
192
+ ```
193
+
194
+ ## 查询客户
195
+
196
+ 按姓名/手机号查:
197
+
198
+ ```bash
199
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
200
+ customer query --json '{"name":"张三","limit":10}'
201
+ ```
202
+
203
+ 查本月生日:
204
+
205
+ ```bash
206
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
207
+ customer query --json '{"birthday_month":3,"limit":20}'
208
+ ```
209
+
210
+ 查 30 天内到期:
211
+
212
+ ```bash
213
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
214
+ customer query --json '{"due_within_days":30,"limit":20}'
215
+ ```
216
+
217
+ 返回结果中会带 Markdown 预览,可直接展示给用户。
218
+
219
+ ## 修改客户
220
+
221
+ 更新前先查,确认命中唯一客户后再执行:
222
+
223
+ ```bash
224
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
225
+ customer update \
226
+ --match-json '{"name":"张三"}' \
227
+ --updates-json '{"primary_phone":"13900139000","dynamic_fields":{"孩子生日":"2018-03-01"}}'
228
+ ```
229
+
230
+ ## 删除客户
231
+
232
+ 删除必须二次确认:
233
+
234
+ ```bash
235
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
236
+ customer delete --match-json '{"name":"张三"}'
237
+ ```
238
+
239
+ ## 提醒
240
+
241
+ 先规划提醒,再决定是否真正注册到 OpenClaw cron。
242
+
243
+ 规划生日和到期提醒:
244
+
245
+ ```bash
246
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
247
+ reminder plan --json '{"kinds":["birthday","due_date"]}'
248
+ ```
249
+
250
+ 确认后创建提醒:
251
+
252
+ ```bash
253
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
254
+ reminder schedule --json '{"kinds":["birthday","due_date"]}'
255
+ ```
256
+
257
+ 单个客户跟进提醒:
258
+
259
+ ```bash
260
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
261
+ reminder plan --json '{"follow_up":{"match":{"name":"张三"},"date":"2026-03-20"}}'
262
+ ```
263
+
264
+ ## 导出
265
+
266
+ 优先导出 CSV,必要时再导出 XLSX:
267
+
268
+ ```bash
269
+ uv run --project "{baseDir}" python -m customer_mgmt_cli \
270
+ export file --output "{baseDir}/data/customers_export.csv"
271
+ ```
@@ -0,0 +1,15 @@
1
+ [project]
2
+ name = "sophnet-customer-management"
3
+ version = "1.0.0"
4
+ description = "Customer management skill core and CLI"
5
+ requires-python = ">=3.10"
6
+ dependencies = [
7
+ "openpyxl>=3.1.0",
8
+ "xlrd>=2.0.1",
9
+ ]
10
+
11
+ [project.scripts]
12
+ customer-mgmt = "customer_mgmt_cli.cli:main"
13
+
14
+ [tool.uv]
15
+ package = true
@@ -0,0 +1,2 @@
1
+ """Customer management CLI package."""
2
+
@@ -0,0 +1,5 @@
1
+ from .cli import main
2
+
3
+
4
+ if __name__ == "__main__":
5
+ raise SystemExit(main())
@@ -0,0 +1,67 @@
1
+ from __future__ import annotations
2
+
3
+ import argparse
4
+ import json
5
+ import sys
6
+
7
+ from customer_mgmt_core.service import CustomerManagementService
8
+
9
+ from .commands import customer, export_file, import_file, reminder, schema
10
+ from .config import CONFIG_FILE, resolve_db_path
11
+
12
+
13
+ def build_parser() -> argparse.ArgumentParser:
14
+ parser = argparse.ArgumentParser(description="客户管理 CLI")
15
+ parser.add_argument(
16
+ "--db",
17
+ default=None,
18
+ help="SQLite 数据库路径;不传时优先读取 ~/.config/sophnet-customer-management/config.env",
19
+ )
20
+ subparsers = parser.add_subparsers(dest="command", required=True)
21
+ schema.register(subparsers)
22
+ import_file.register(subparsers)
23
+ customer.register(subparsers)
24
+ reminder.register(subparsers)
25
+ export_file.register(subparsers)
26
+ return parser
27
+
28
+
29
+ def main() -> int:
30
+ parser = build_parser()
31
+ args = parser.parse_args()
32
+ db_path = resolve_db_path(args.db)
33
+ service = CustomerManagementService(db_path)
34
+ try:
35
+ result = args.handler(args, service)
36
+ except Exception as exc: # noqa: BLE001
37
+ print(
38
+ json.dumps(
39
+ {
40
+ "ok": False,
41
+ "db_path": str(db_path),
42
+ "config_file": str(CONFIG_FILE),
43
+ "error": str(exc),
44
+ },
45
+ ensure_ascii=False,
46
+ indent=2,
47
+ )
48
+ )
49
+ return 1
50
+
51
+ print(
52
+ json.dumps(
53
+ {
54
+ "ok": True,
55
+ "db_path": str(db_path),
56
+ "config_file": str(CONFIG_FILE),
57
+ "result": result,
58
+ },
59
+ ensure_ascii=False,
60
+ indent=2,
61
+ )
62
+ )
63
+ return 0
64
+
65
+
66
+ if __name__ == "__main__":
67
+ sys.exit(main())
@@ -0,0 +1,60 @@
1
+ from __future__ import annotations
2
+
3
+ import json
4
+
5
+
6
+ def register(subparsers) -> None:
7
+ customer_parser = subparsers.add_parser("customer", help="客户增删改查")
8
+ customer_subparsers = customer_parser.add_subparsers(dest="customer_action", required=True)
9
+
10
+ add_parser = customer_subparsers.add_parser("add", help="新增客户")
11
+ add_parser.add_argument("--json", dest="payload_json", required=True)
12
+ add_parser.set_defaults(handler=handle_add)
13
+
14
+ update_parser = customer_subparsers.add_parser("update", help="更新客户")
15
+ update_parser.add_argument("--match-json", required=True)
16
+ update_parser.add_argument("--updates-json", required=True)
17
+ update_parser.set_defaults(handler=handle_update)
18
+
19
+ delete_parser = customer_subparsers.add_parser("delete", help="删除客户")
20
+ delete_parser.add_argument("--match-json", required=True)
21
+ delete_parser.set_defaults(handler=handle_delete)
22
+
23
+ query_parser = customer_subparsers.add_parser("query", help="查询客户")
24
+ query_parser.add_argument("--json", dest="payload_json", required=True)
25
+ query_parser.set_defaults(handler=handle_query)
26
+
27
+ record_add_parser = customer_subparsers.add_parser("record-add", help="新增业务记录")
28
+ record_add_parser.add_argument("--json", dest="payload_json", required=True)
29
+ record_add_parser.set_defaults(handler=handle_record_add)
30
+
31
+ record_query_parser = customer_subparsers.add_parser("record-query", help="查询业务记录")
32
+ record_query_parser.add_argument("--json", dest="payload_json", required=True)
33
+ record_query_parser.set_defaults(handler=handle_record_query)
34
+
35
+
36
+ def handle_add(args, service):
37
+ return service.add_customer(json.loads(args.payload_json))
38
+
39
+
40
+ def handle_update(args, service):
41
+ return service.update_customer(
42
+ json.loads(args.match_json),
43
+ json.loads(args.updates_json),
44
+ )
45
+
46
+
47
+ def handle_delete(args, service):
48
+ return service.delete_customer(json.loads(args.match_json))
49
+
50
+
51
+ def handle_query(args, service):
52
+ return service.query_customers(json.loads(args.payload_json))
53
+
54
+
55
+ def handle_record_add(args, service):
56
+ return service.add_record(json.loads(args.payload_json))
57
+
58
+
59
+ def handle_record_query(args, service):
60
+ return service.query_records(json.loads(args.payload_json))
@@ -0,0 +1,18 @@
1
+ from __future__ import annotations
2
+
3
+ import json
4
+
5
+
6
+ def register(subparsers) -> None:
7
+ export_parser = subparsers.add_parser("export", help="导出客户数据")
8
+ export_subparsers = export_parser.add_subparsers(dest="export_action", required=True)
9
+
10
+ file_parser = export_subparsers.add_parser("file", help="导出到文件")
11
+ file_parser.add_argument("--output", required=True)
12
+ file_parser.add_argument("--json", dest="payload_json")
13
+ file_parser.set_defaults(handler=handle_export_file)
14
+
15
+
16
+ def handle_export_file(args, service):
17
+ payload = json.loads(args.payload_json) if args.payload_json else {}
18
+ return service.export_customers(args.output, payload)
@@ -0,0 +1,15 @@
1
+ from __future__ import annotations
2
+
3
+
4
+ def register(subparsers) -> None:
5
+ import_parser = subparsers.add_parser("import", help="导入客户文件")
6
+ import_subparsers = import_parser.add_subparsers(dest="import_action", required=True)
7
+
8
+ file_parser = import_subparsers.add_parser("file", help="导入文件")
9
+ file_parser.add_argument("--path", required=True)
10
+ file_parser.add_argument("--commit", action="store_true")
11
+ file_parser.set_defaults(handler=handle_import_file)
12
+
13
+
14
+ def handle_import_file(args, service):
15
+ return service.import_file(args.path, commit=args.commit)
@@ -0,0 +1,26 @@
1
+ from __future__ import annotations
2
+
3
+ import json
4
+
5
+
6
+ def register(subparsers) -> None:
7
+ reminder_parser = subparsers.add_parser("reminder", help="提醒规划与创建")
8
+ reminder_subparsers = reminder_parser.add_subparsers(dest="reminder_action", required=True)
9
+
10
+ plan_parser = reminder_subparsers.add_parser("plan", help="生成提醒计划")
11
+ plan_parser.add_argument("--json", dest="payload_json", required=True)
12
+ plan_parser.set_defaults(handler=handle_plan)
13
+
14
+ schedule_parser = reminder_subparsers.add_parser("schedule", help="生成并注册提醒")
15
+ schedule_parser.add_argument("--json", dest="payload_json", required=True)
16
+ schedule_parser.set_defaults(handler=handle_schedule)
17
+
18
+
19
+ def handle_plan(args, service):
20
+ return service.plan_reminders(json.loads(args.payload_json))
21
+
22
+
23
+ def handle_schedule(args, service):
24
+ payload = json.loads(args.payload_json)
25
+ payload["schedule"] = True
26
+ return service.plan_reminders(payload)
@@ -0,0 +1,28 @@
1
+ from __future__ import annotations
2
+
3
+ import json
4
+
5
+
6
+ def register(subparsers) -> None:
7
+ schema_parser = subparsers.add_parser("schema", help="查看或演进字段定义")
8
+ schema_subparsers = schema_parser.add_subparsers(dest="schema_action", required=True)
9
+
10
+ inspect_parser = schema_subparsers.add_parser("inspect", help="查看当前 schema")
11
+ inspect_parser.set_defaults(handler=handle_inspect)
12
+
13
+ evolve_parser = schema_subparsers.add_parser("evolve", help="新增动态字段")
14
+ evolve_parser.add_argument("--labels", nargs="*", default=[])
15
+ evolve_parser.add_argument("--json", dest="payload_json")
16
+ evolve_parser.set_defaults(handler=handle_evolve)
17
+
18
+
19
+ def handle_inspect(args, service):
20
+ return service.inspect_schema()
21
+
22
+
23
+ def handle_evolve(args, service):
24
+ labels = list(args.labels)
25
+ if args.payload_json:
26
+ payload = json.loads(args.payload_json)
27
+ labels.extend(payload.get("labels", []))
28
+ return service.evolve_schema(labels)
@@ -0,0 +1,54 @@
1
+ from __future__ import annotations
2
+
3
+ import os
4
+ from pathlib import Path
5
+
6
+
7
+ CONFIG_DIR = Path.home() / ".config" / "sophnet-customer-management"
8
+ CONFIG_FILE = CONFIG_DIR / "config.env"
9
+ DEFAULT_DB_FILE = CONFIG_DIR / "customer_mgmt.sqlite3"
10
+ DB_PATH_KEY = "CUSTOMER_MGMT_DB_PATH"
11
+
12
+
13
+ def resolve_db_path(cli_db: str | None) -> Path:
14
+ if cli_db:
15
+ db_path = Path(cli_db).expanduser()
16
+ persist_db_path(db_path)
17
+ return db_path
18
+
19
+ env_db = load_env_value(DB_PATH_KEY)
20
+ if env_db:
21
+ return Path(env_db).expanduser()
22
+
23
+ default_db = DEFAULT_DB_FILE
24
+ persist_db_path(default_db)
25
+ return default_db
26
+
27
+
28
+ def persist_db_path(db_path: Path) -> None:
29
+ CONFIG_DIR.mkdir(parents=True, exist_ok=True)
30
+ entries = load_env_entries()
31
+ entries[DB_PATH_KEY] = str(db_path.expanduser().resolve())
32
+ lines = ["# sophnet-customer-management runtime config"]
33
+ lines.extend(f"{key}={value}" for key, value in sorted(entries.items()))
34
+ CONFIG_FILE.write_text("\n".join(lines) + "\n", encoding="utf-8")
35
+
36
+
37
+ def load_env_value(key: str) -> str | None:
38
+ value = load_env_entries().get(key)
39
+ if value:
40
+ return value
41
+ return os.environ.get(key)
42
+
43
+
44
+ def load_env_entries() -> dict[str, str]:
45
+ if not CONFIG_FILE.exists():
46
+ return {}
47
+ entries: dict[str, str] = {}
48
+ for raw_line in CONFIG_FILE.read_text(encoding="utf-8").splitlines():
49
+ line = raw_line.strip()
50
+ if not line or line.startswith("#") or "=" not in line:
51
+ continue
52
+ key, value = line.split("=", 1)
53
+ entries[key.strip()] = value.strip()
54
+ return entries
@@ -0,0 +1,2 @@
1
+ """Customer management core package."""
2
+
@@ -0,0 +1,85 @@
1
+ from __future__ import annotations
2
+
3
+ import csv
4
+ import json
5
+ from pathlib import Path
6
+ from typing import Any
7
+
8
+ import openpyxl
9
+
10
+
11
+ def flatten_customers(customers: list[dict[str, Any]]) -> list[dict[str, Any]]:
12
+ rows: list[dict[str, Any]] = []
13
+ for customer in customers:
14
+ row = {
15
+ "客户ID": customer["customer_id"],
16
+ "姓名": customer.get("name"),
17
+ "手机号": customer.get("primary_phone"),
18
+ "生日": customer.get("birthday"),
19
+ "标签": ",".join(customer.get("tags", [])),
20
+ "备注": customer.get("notes"),
21
+ "最近更新时间": customer.get("updated_at"),
22
+ }
23
+ for field_key, item in customer.get("dynamic_field_map", {}).items():
24
+ row[item.get("label") or field_key] = item.get("displayValue") or item.get("value")
25
+ rows.append(row)
26
+ return rows
27
+
28
+
29
+ def export_customers(customers: list[dict[str, Any]], output_path: str | Path) -> dict[str, Any]:
30
+ path = Path(output_path)
31
+ path.parent.mkdir(parents=True, exist_ok=True)
32
+ rows = flatten_customers(customers)
33
+ if path.suffix.lower() == ".csv":
34
+ _write_csv(rows, path)
35
+ elif path.suffix.lower() == ".xlsx":
36
+ _write_xlsx(rows, path)
37
+ elif path.suffix.lower() == ".json":
38
+ path.write_text(json.dumps(rows, ensure_ascii=False, indent=2), encoding="utf-8")
39
+ else:
40
+ raise ValueError("仅支持导出为 .csv / .xlsx / .json")
41
+ return {"output_path": str(path), "row_count": len(rows)}
42
+
43
+
44
+ def preview_as_markdown(customers: list[dict[str, Any]], limit: int = 20) -> str:
45
+ rows = flatten_customers(customers)[:limit]
46
+ if not rows:
47
+ return "暂无客户数据。"
48
+ headers = list(rows[0].keys())
49
+ lines = [
50
+ "| " + " | ".join(headers) + " |",
51
+ "| " + " | ".join(["---"] * len(headers)) + " |",
52
+ ]
53
+ for row in rows:
54
+ lines.append("| " + " | ".join(str(row.get(header, "") or "") for header in headers) + " |")
55
+ return "\n".join(lines)
56
+
57
+
58
+ def _write_csv(rows: list[dict[str, Any]], path: Path) -> None:
59
+ headers = _collect_headers(rows)
60
+ with path.open("w", encoding="utf-8-sig", newline="") as file:
61
+ writer = csv.DictWriter(file, fieldnames=headers)
62
+ writer.writeheader()
63
+ writer.writerows(rows)
64
+
65
+
66
+ def _write_xlsx(rows: list[dict[str, Any]], path: Path) -> None:
67
+ workbook = openpyxl.Workbook()
68
+ sheet = workbook.active
69
+ sheet.title = "客户数据"
70
+ headers = _collect_headers(rows)
71
+ sheet.append(headers)
72
+ for row in rows:
73
+ sheet.append([row.get(header, "") for header in headers])
74
+ workbook.save(path)
75
+
76
+
77
+ def _collect_headers(rows: list[dict[str, Any]]) -> list[str]:
78
+ headers: list[str] = []
79
+ seen: set[str] = set()
80
+ for row in rows:
81
+ for key in row.keys():
82
+ if key not in seen:
83
+ seen.add(key)
84
+ headers.append(key)
85
+ return headers