@qingflow-tech/qingflow-app-user-mcp 1.0.11 → 1.0.12

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 (86) hide show
  1. package/README.md +9 -3
  2. package/docs/local-agent-install.md +54 -3
  3. package/entry_point.py +1 -1
  4. package/npm/bin/qingflow-skills.mjs +5 -0
  5. package/npm/lib/runtime.mjs +304 -13
  6. package/npm/scripts/postinstall.mjs +1 -5
  7. package/package.json +3 -2
  8. package/pyproject.toml +1 -1
  9. package/skills/qingflow-app-builder/SKILL.md +255 -0
  10. package/skills/qingflow-app-builder/agents/openai.yaml +4 -0
  11. package/skills/qingflow-app-builder/references/create-app.md +149 -0
  12. package/skills/qingflow-app-builder/references/environments.md +63 -0
  13. package/skills/qingflow-app-builder/references/flow-actors-and-permissions.md +123 -0
  14. package/skills/qingflow-app-builder/references/gotchas.md +107 -0
  15. package/skills/qingflow-app-builder/references/match-rules.md +114 -0
  16. package/skills/qingflow-app-builder/references/public-surface-sync.md +75 -0
  17. package/skills/qingflow-app-builder/references/solution-playbooks.md +52 -0
  18. package/skills/qingflow-app-builder/references/tool-selection.md +99 -0
  19. package/skills/qingflow-app-builder/references/update-flow.md +158 -0
  20. package/skills/qingflow-app-builder/references/update-layout.md +68 -0
  21. package/skills/qingflow-app-builder/references/update-schema.md +72 -0
  22. package/skills/qingflow-app-builder/references/update-views.md +284 -0
  23. package/skills/qingflow-app-builder-code-integrations/SKILL.md +137 -0
  24. package/skills/qingflow-app-builder-code-integrations/agents/openai.yaml +4 -0
  25. package/skills/qingflow-app-builder-code-integrations/references/code-block.md +66 -0
  26. package/skills/qingflow-app-builder-code-integrations/references/q-linker.md +77 -0
  27. package/skills/qingflow-app-user/SKILL.md +12 -11
  28. package/skills/qingflow-app-user/references/data-gotchas.md +2 -2
  29. package/skills/qingflow-app-user/references/public-surface-sync.md +3 -3
  30. package/skills/qingflow-app-user/references/record-patterns.md +5 -5
  31. package/skills/qingflow-app-user/references/workflow-usage.md +4 -5
  32. package/skills/qingflow-mcp-setup/SKILL.md +113 -0
  33. package/skills/qingflow-mcp-setup/agents/openai.yaml +4 -0
  34. package/skills/qingflow-mcp-setup/references/claude-desktop.md +34 -0
  35. package/skills/qingflow-mcp-setup/references/environments.md +62 -0
  36. package/skills/qingflow-mcp-setup/references/generic-stdio.md +32 -0
  37. package/skills/qingflow-mcp-setup/scripts/check_local_server.sh +38 -0
  38. package/skills/qingflow-record-analysis/SKILL.md +6 -7
  39. package/skills/qingflow-record-analysis/manifest.yaml +10 -0
  40. package/skills/qingflow-record-delete/SKILL.md +5 -3
  41. package/skills/qingflow-record-import/SKILL.md +6 -2
  42. package/skills/qingflow-record-insert/SKILL.md +48 -4
  43. package/skills/qingflow-record-insert/manifest.yaml +6 -0
  44. package/skills/qingflow-record-update/SKILL.md +36 -24
  45. package/skills/qingflow-task-ops/SKILL.md +25 -25
  46. package/skills/qingflow-task-ops/references/environments.md +0 -1
  47. package/skills/qingflow-task-ops/references/workflow-usage.md +4 -6
  48. package/src/qingflow_mcp/__main__.py +6 -2
  49. package/src/qingflow_mcp/builder_facade/service.py +1488 -288
  50. package/src/qingflow_mcp/cli/commands/builder.py +2 -2
  51. package/src/qingflow_mcp/cli/commands/exports.py +2 -2
  52. package/src/qingflow_mcp/cli/commands/imports.py +1 -1
  53. package/src/qingflow_mcp/cli/commands/record.py +39 -11
  54. package/src/qingflow_mcp/cli/context.py +0 -3
  55. package/src/qingflow_mcp/cli/formatters.py +206 -7
  56. package/src/qingflow_mcp/cli/main.py +47 -3
  57. package/src/qingflow_mcp/errors.py +43 -2
  58. package/src/qingflow_mcp/public_surface.py +21 -15
  59. package/src/qingflow_mcp/response_trim.py +68 -13
  60. package/src/qingflow_mcp/server.py +11 -9
  61. package/src/qingflow_mcp/server_app_builder.py +3 -2
  62. package/src/qingflow_mcp/server_app_user.py +15 -13
  63. package/src/qingflow_mcp/solution/executor.py +112 -15
  64. package/src/qingflow_mcp/tools/ai_builder_tools.py +36 -11
  65. package/src/qingflow_mcp/tools/app_tools.py +184 -43
  66. package/src/qingflow_mcp/tools/approval_tools.py +196 -34
  67. package/src/qingflow_mcp/tools/auth_tools.py +92 -16
  68. package/src/qingflow_mcp/tools/code_block_tools.py +296 -39
  69. package/src/qingflow_mcp/tools/custom_button_tools.py +64 -10
  70. package/src/qingflow_mcp/tools/directory_tools.py +236 -72
  71. package/src/qingflow_mcp/tools/export_tools.py +230 -33
  72. package/src/qingflow_mcp/tools/file_tools.py +7 -3
  73. package/src/qingflow_mcp/tools/import_tools.py +293 -40
  74. package/src/qingflow_mcp/tools/navigation_tools.py +91 -12
  75. package/src/qingflow_mcp/tools/package_tools.py +118 -6
  76. package/src/qingflow_mcp/tools/portal_tools.py +39 -3
  77. package/src/qingflow_mcp/tools/qingbi_report_tools.py +116 -7
  78. package/src/qingflow_mcp/tools/record_tools.py +1042 -338
  79. package/src/qingflow_mcp/tools/resource_read_tools.py +188 -39
  80. package/src/qingflow_mcp/tools/role_tools.py +80 -9
  81. package/src/qingflow_mcp/tools/solution_tools.py +57 -15
  82. package/src/qingflow_mcp/tools/task_context_tools.py +569 -119
  83. package/src/qingflow_mcp/tools/task_tools.py +113 -29
  84. package/src/qingflow_mcp/tools/view_tools.py +106 -3
  85. package/src/qingflow_mcp/tools/workflow_tools.py +17 -1
  86. package/src/qingflow_mcp/tools/workspace_tools.py +71 -3
package/README.md CHANGED
@@ -3,13 +3,13 @@
3
3
  Install:
4
4
 
5
5
  ```bash
6
- npm install @qingflow-tech/qingflow-app-user-mcp@1.0.11
6
+ npm install @qingflow-tech/qingflow-app-user-mcp@1.0.12
7
7
  ```
8
8
 
9
9
  Run:
10
10
 
11
11
  ```bash
12
- npx -y -p @qingflow-tech/qingflow-app-user-mcp@1.0.11 qingflow-app-user-mcp
12
+ npx -y -p @qingflow-tech/qingflow-app-user-mcp@1.0.12 qingflow-app-user-mcp
13
13
  ```
14
14
 
15
15
  Environment:
@@ -23,6 +23,9 @@ This package bootstraps a local Python runtime on first install and then starts
23
23
  Bundled skills:
24
24
 
25
25
  - `skills/qingflow-app-user`
26
+ - `skills/qingflow-mcp-setup`
27
+ - `skills/qingflow-app-builder`
28
+ - `skills/qingflow-app-builder-code-integrations`
26
29
  - `skills/qingflow-record-insert`
27
30
  - `skills/qingflow-record-update`
28
31
  - `skills/qingflow-record-delete`
@@ -33,5 +36,8 @@ Bundled skills:
33
36
  Note:
34
37
 
35
38
  - The skill files are included in the npm package.
36
- - On install, the package copies them to `$CODEX_HOME/skills` (or `~/.codex/skills` if `CODEX_HOME` is unset).
39
+ - Installing the npm package does not overwrite agent skills automatically.
40
+ - To mount bundled skills from an installed package, run `qingflow-app-user-mcp-skills install --agent codex --scope user`.
41
+ - For one-shot `npx -p` installs, prefer `--copy` because symlinks would point into the npm execution cache.
42
+ - Use `qingflow-app-user-mcp-skills list` to inspect bundled skills. The installer defaults to symlinks for stable package installs and refuses to overwrite existing skills unless `--force` is provided.
37
43
  - If a stdio MCP client reports `Transport closed`, delete `.npm-python`, reinstall the package, and make sure CLI/user/builder packages are on the same version. The stdio entrypoints refuse runtime bootstrap so install logs never corrupt MCP stdout.
@@ -101,7 +101,51 @@ npm install /absolute/path/to/dist/npm/qingflow-tech-qingflow-app-builder-mcp-<v
101
101
  1. 创建 `.npm-python/`
102
102
  2. 在其中建立 Python 虚拟环境
103
103
  3. 执行 `pip install .`
104
- 4. 在安装位置暴露 `qingflow`、`qingflow-app-user-mcp`、`qingflow-app-builder-mcp` 命令
104
+ 4. 在安装位置暴露对应入口:CLI 包暴露 `qingflow` / `qingflow-skills`;app-user 包暴露 `qingflow-app-user-mcp` / `qingflow-app-user-mcp-skills`;app-builder 包暴露 `qingflow-app-builder-mcp` / `qingflow-app-builder-mcp-skills`
105
+ 5. 携带 `skills/<skill-name>/SKILL.md`,但不在 `postinstall` 阶段自动覆盖本机 agent skills
106
+
107
+ ## Skills 挂载
108
+
109
+ 显式查看包内 skills:
110
+
111
+ ```bash
112
+ qingflow-skills list
113
+ ```
114
+
115
+ 独立 MCP split 包使用包专属 skills 命令,避免全局安装多个包时发生 bin 覆盖:
116
+
117
+ ```bash
118
+ qingflow-app-user-mcp-skills list
119
+ qingflow-app-builder-mcp-skills list
120
+ ```
121
+
122
+ 显式挂载到 Codex 用户目录:
123
+
124
+ ```bash
125
+ qingflow-skills install --agent codex --scope user
126
+ ```
127
+
128
+ 如果通过一次性 `npx -p <package>` 执行安装,请加 `--copy`,避免 symlink 指向 npm 临时执行缓存:
129
+
130
+ ```bash
131
+ npx -y -p @josephyan/qingflow-cli qingflow-skills install --agent codex --scope user --copy
132
+ ```
133
+
134
+ 也可以挂载到项目级 agent 目录:
135
+
136
+ ```bash
137
+ qingflow-skills install --agent claude-code --scope project
138
+ qingflow-skills install --agent cursor --scope project --copy
139
+ qingflow-skills install --agent all --scope project
140
+ ```
141
+
142
+ 默认行为:
143
+
144
+ - `--mode symlink`:使用 symlink,让 npm 包版本成为单一来源
145
+ - `--scope user`:安装到用户级 agent skills 目录
146
+ - `--agent codex`:目标 agent 为 Codex
147
+ - 不覆盖已有同名 skill;需要替换时显式加 `--force`
148
+ - 每次安装会在目标 skills 目录下写入 `.qingflow-skill-sources/<skill>.json`,记录 package、version、source、destination、agent、scope、mode
105
149
 
106
150
  ## 本地验证
107
151
 
@@ -110,24 +154,31 @@ npm install /absolute/path/to/dist/npm/qingflow-tech-qingflow-app-builder-mcp-<v
110
154
  ```bash
111
155
  cd qingflow-support/mcp-server
112
156
  node ./npm/bin/qingflow.mjs --help
157
+ node ./npm/bin/qingflow-skills.mjs list
113
158
  node ./npm/bin/qingflow-app-user-mcp.mjs
114
159
  node ./npm/bin/qingflow-app-builder-mcp.mjs
115
160
  ```
116
161
 
117
- 如果你是全局安装:
162
+ 如果你是全局安装对应包:
118
163
 
119
164
  ```bash
120
165
  qingflow --help
166
+ qingflow-skills list
121
167
  qingflow-app-user-mcp
168
+ qingflow-app-user-mcp-skills list
122
169
  qingflow-app-builder-mcp
170
+ qingflow-app-builder-mcp-skills list
123
171
  ```
124
172
 
125
- 如果你是把包安装到了某个本地 agent workspace,命令通常位于:
173
+ 如果你是把包安装到了某个本地 agent workspace,安装对应包后命令通常位于:
126
174
 
127
175
  ```bash
128
176
  /absolute/path/to/agent-workspace/node_modules/.bin/qingflow
177
+ /absolute/path/to/agent-workspace/node_modules/.bin/qingflow-skills
129
178
  /absolute/path/to/agent-workspace/node_modules/.bin/qingflow-app-user-mcp
179
+ /absolute/path/to/agent-workspace/node_modules/.bin/qingflow-app-user-mcp-skills
130
180
  /absolute/path/to/agent-workspace/node_modules/.bin/qingflow-app-builder-mcp
181
+ /absolute/path/to/agent-workspace/node_modules/.bin/qingflow-app-builder-mcp-skills
131
182
  ```
132
183
 
133
184
  如果你是从 tgz 安装到某个空目录,命令通常位于:
package/entry_point.py CHANGED
@@ -7,7 +7,7 @@ src_path = os.path.abspath(os.path.join(os.path.dirname(__file__), "src"))
7
7
  if src_path not in sys.path:
8
8
  sys.path.insert(0, src_path)
9
9
 
10
- from qingflow_mcp.server import main
10
+ from qingflow_mcp.server_app_user import main
11
11
 
12
12
  if __name__ == "__main__":
13
13
  main()
@@ -0,0 +1,5 @@
1
+ #!/usr/bin/env node
2
+ import { getPackageRoot, runSkillsCli } from "../lib/runtime.mjs";
3
+
4
+ const packageRoot = getPackageRoot(import.meta.url);
5
+ runSkillsCli(packageRoot, process.argv.slice(2));
@@ -43,29 +43,320 @@ export function getCodexHome() {
43
43
  return path.join(home, ".codex");
44
44
  }
45
45
 
46
- export function installBundledSkills(packageRoot) {
46
+ function getHomeDir() {
47
+ const home = process.env.HOME || process.env.USERPROFILE;
48
+ if (!home) {
49
+ throw new Error("Cannot resolve user home because HOME is not set.");
50
+ }
51
+ return home;
52
+ }
53
+
54
+ function getAgentSkillDest(agent, scope, cwd = process.cwd()) {
55
+ const normalizedAgent = agent.trim().toLowerCase();
56
+ const normalizedScope = scope.trim().toLowerCase();
57
+ const home = getHomeDir();
58
+ if (!["user", "project"].includes(normalizedScope)) {
59
+ throw new Error(`Unsupported skills scope '${scope}'. Expected 'user' or 'project'.`);
60
+ }
61
+
62
+ const projectRoots = {
63
+ codex: [cwd, ".codex", "skills"],
64
+ claude: [cwd, ".claude", "skills"],
65
+ "claude-code": [cwd, ".claude", "skills"],
66
+ cursor: [cwd, ".cursor", "skills"],
67
+ generic: [cwd, ".agents", "skills"],
68
+ };
69
+ const userRoots = {
70
+ codex: [process.env.CODEX_HOME?.trim() || path.join(home, ".codex"), "skills"],
71
+ claude: [home, ".claude", "skills"],
72
+ "claude-code": [home, ".claude", "skills"],
73
+ cursor: [home, ".cursor", "skills"],
74
+ generic: [home, ".agents", "skills"],
75
+ };
76
+
77
+ const roots = normalizedScope === "project" ? projectRoots : userRoots;
78
+ const parts = roots[normalizedAgent];
79
+ if (!parts) {
80
+ throw new Error(`Unsupported skills agent '${agent}'. Expected one of: codex, claude, claude-code, cursor, generic.`);
81
+ }
82
+ return path.resolve(...parts);
83
+ }
84
+
85
+ export function listBundledSkills(packageRoot) {
47
86
  const skillsSrc = path.join(packageRoot, "skills");
48
87
  if (!fs.existsSync(skillsSrc)) {
49
- return { installed: [], skipped: true, destination: null };
88
+ return [];
50
89
  }
51
90
 
52
- const codexHome = getCodexHome();
53
- const skillsDestRoot = path.join(codexHome, "skills");
54
- fs.mkdirSync(skillsDestRoot, { recursive: true });
91
+ return fs
92
+ .readdirSync(skillsSrc, { withFileTypes: true })
93
+ .filter((entry) => entry.isDirectory() && fs.existsSync(path.join(skillsSrc, entry.name, "SKILL.md")))
94
+ .map((entry) => entry.name)
95
+ .sort();
96
+ }
97
+
98
+ function sameRealPath(a, b) {
99
+ try {
100
+ return fs.realpathSync(a) === fs.realpathSync(b);
101
+ } catch {
102
+ return false;
103
+ }
104
+ }
55
105
 
56
- const installed = [];
57
- for (const entry of fs.readdirSync(skillsSrc, { withFileTypes: true })) {
58
- if (!entry.isDirectory()) {
59
- continue;
106
+ function installOneSkill(src, dest, { force, mode }) {
107
+ let existingStat = null;
108
+ try {
109
+ existingStat = fs.lstatSync(dest);
110
+ } catch (error) {
111
+ if (error.code !== "ENOENT") {
112
+ throw error;
113
+ }
114
+ }
115
+ if (existingStat) {
116
+ if (existingStat.isSymbolicLink() && sameRealPath(dest, src)) {
117
+ return { status: "unchanged" };
118
+ }
119
+ if (!force) {
120
+ return { status: "conflict" };
60
121
  }
61
- const src = path.join(skillsSrc, entry.name);
62
- const dest = path.join(skillsDestRoot, entry.name);
63
122
  fs.rmSync(dest, { recursive: true, force: true });
123
+ }
124
+
125
+ if (mode === "copy") {
64
126
  fs.cpSync(src, dest, { recursive: true });
65
- installed.push(entry.name);
127
+ return { status: "installed" };
128
+ }
129
+
130
+ fs.symlinkSync(src, dest, "dir");
131
+ return { status: "installed" };
132
+ }
133
+
134
+ function writeSkillProvenance(skillsDestRoot, skillName, payload) {
135
+ const provenanceRoot = path.join(skillsDestRoot, ".qingflow-skill-sources");
136
+ fs.mkdirSync(provenanceRoot, { recursive: true });
137
+ fs.writeFileSync(path.join(provenanceRoot, `${skillName}.json`), `${JSON.stringify(payload, null, 2)}\n`);
138
+ }
139
+
140
+ export function installBundledSkills(
141
+ packageRoot,
142
+ {
143
+ agent = "codex",
144
+ scope = "user",
145
+ mode = "symlink",
146
+ force = false,
147
+ skills = [],
148
+ cwd = process.cwd(),
149
+ } = {},
150
+ ) {
151
+ const skillsSrc = path.join(packageRoot, "skills");
152
+ if (!fs.existsSync(skillsSrc)) {
153
+ return { installed: [], unchanged: [], conflicts: [], skipped: true, destination: null };
66
154
  }
67
155
 
68
- return { installed, skipped: false, destination: skillsDestRoot };
156
+ const validModes = new Set(["symlink", "copy"]);
157
+ if (!validModes.has(mode)) {
158
+ throw new Error(`Unsupported skills install mode '${mode}'. Expected 'symlink' or 'copy'.`);
159
+ }
160
+
161
+ const available = listBundledSkills(packageRoot);
162
+ const wanted = skills.length ? skills : available;
163
+ const unknown = wanted.filter((skillName) => !available.includes(skillName));
164
+ if (unknown.length) {
165
+ throw new Error(`Unknown bundled skill(s): ${unknown.join(", ")}. Available: ${available.join(", ")}`);
166
+ }
167
+
168
+ const skillsDestRoot = getAgentSkillDest(agent, scope, cwd);
169
+ fs.mkdirSync(skillsDestRoot, { recursive: true });
170
+
171
+ const packageJson = JSON.parse(fs.readFileSync(path.join(packageRoot, "package.json"), "utf8"));
172
+ const result = {
173
+ installed: [],
174
+ unchanged: [],
175
+ conflicts: [],
176
+ skipped: false,
177
+ destination: skillsDestRoot,
178
+ agent,
179
+ scope,
180
+ mode,
181
+ };
182
+
183
+ for (const skillName of wanted) {
184
+ const src = path.join(skillsSrc, skillName);
185
+ const dest = path.join(skillsDestRoot, skillName);
186
+ const installResult = installOneSkill(src, dest, { force, mode });
187
+ result[installResult.status === "conflict" ? "conflicts" : installResult.status].push(skillName);
188
+ if (installResult.status !== "conflict") {
189
+ writeSkillProvenance(skillsDestRoot, skillName, {
190
+ package: packageJson.name ?? null,
191
+ version: packageJson.version ?? null,
192
+ skill: skillName,
193
+ source: src,
194
+ destination: dest,
195
+ agent,
196
+ scope,
197
+ mode,
198
+ installed_at: new Date().toISOString(),
199
+ });
200
+ }
201
+ }
202
+
203
+ return result;
204
+ }
205
+
206
+ function parseSkillsCliArgs(args) {
207
+ const parsed = {
208
+ command: "help",
209
+ agent: "codex",
210
+ scope: "user",
211
+ mode: "symlink",
212
+ force: false,
213
+ json: false,
214
+ skills: [],
215
+ };
216
+ const rest = [...args];
217
+ if (rest.length && !rest[0].startsWith("-")) {
218
+ parsed.command = rest.shift();
219
+ }
220
+ for (let index = 0; index < rest.length; index += 1) {
221
+ const arg = rest[index];
222
+ const next = () => {
223
+ index += 1;
224
+ if (index >= rest.length) {
225
+ throw new Error(`Missing value for ${arg}`);
226
+ }
227
+ return rest[index];
228
+ };
229
+ switch (arg) {
230
+ case "--agent":
231
+ case "-a":
232
+ parsed.agent = next();
233
+ break;
234
+ case "--scope":
235
+ case "-s":
236
+ parsed.scope = next();
237
+ break;
238
+ case "--mode":
239
+ case "-m":
240
+ parsed.mode = next();
241
+ break;
242
+ case "--skill":
243
+ parsed.skills.push(next());
244
+ break;
245
+ case "--all":
246
+ parsed.agent = "all";
247
+ break;
248
+ case "--force":
249
+ case "-f":
250
+ parsed.force = true;
251
+ break;
252
+ case "--copy":
253
+ parsed.mode = "copy";
254
+ break;
255
+ case "--json":
256
+ parsed.json = true;
257
+ break;
258
+ case "--help":
259
+ case "-h":
260
+ parsed.command = "help";
261
+ break;
262
+ default:
263
+ if (arg.startsWith("-")) {
264
+ throw new Error(`Unknown option ${arg}`);
265
+ }
266
+ parsed.skills.push(arg);
267
+ break;
268
+ }
269
+ }
270
+ return parsed;
271
+ }
272
+
273
+ function printSkillsHelp() {
274
+ console.log(`Qingflow bundled skills
275
+
276
+ Usage:
277
+ qingflow-skills list [--json]
278
+ qingflow-skills install [--agent codex|claude|claude-code|cursor|generic|all] [--scope user|project] [--mode symlink|copy] [--skill name] [--force] [--json]
279
+
280
+ Examples:
281
+ qingflow-skills list
282
+ qingflow-skills install --agent codex --scope user
283
+ qingflow-skills install --agent claude-code --scope project --copy
284
+ qingflow-skills install --agent all --scope project
285
+
286
+ Defaults:
287
+ --agent codex
288
+ --scope user
289
+ --mode symlink
290
+
291
+ Existing skills are not overwritten unless --force is provided.`);
292
+ }
293
+
294
+ function printInstallSummary(result) {
295
+ console.log(`[qingflow-skills] destination: ${result.destination}`);
296
+ if (result.installed.length) {
297
+ console.log(`[qingflow-skills] installed: ${result.installed.join(", ")}`);
298
+ }
299
+ if (result.unchanged.length) {
300
+ console.log(`[qingflow-skills] unchanged: ${result.unchanged.join(", ")}`);
301
+ }
302
+ if (result.conflicts.length) {
303
+ console.log(`[qingflow-skills] conflicts: ${result.conflicts.join(", ")}`);
304
+ console.log("[qingflow-skills] Re-run with --force to replace conflicting skills.");
305
+ }
306
+ }
307
+
308
+ export function runSkillsCli(packageRoot, args) {
309
+ let parsed;
310
+ try {
311
+ parsed = parseSkillsCliArgs(args);
312
+ if (parsed.command === "help") {
313
+ printSkillsHelp();
314
+ return;
315
+ }
316
+ if (parsed.command === "list") {
317
+ const skills = listBundledSkills(packageRoot);
318
+ if (parsed.json) {
319
+ console.log(JSON.stringify({ skills }, null, 2));
320
+ } else {
321
+ for (const skill of skills) {
322
+ console.log(skill);
323
+ }
324
+ }
325
+ return;
326
+ }
327
+ if (parsed.command !== "install") {
328
+ throw new Error(`Unknown qingflow skills command '${parsed.command}'.`);
329
+ }
330
+
331
+ const agents = parsed.agent === "all" ? ["codex", "claude-code", "cursor", "generic"] : [parsed.agent];
332
+ const results = agents.map((agent) =>
333
+ installBundledSkills(packageRoot, {
334
+ agent,
335
+ scope: parsed.scope,
336
+ mode: parsed.mode,
337
+ force: parsed.force,
338
+ skills: parsed.skills,
339
+ }),
340
+ );
341
+ if (parsed.json) {
342
+ console.log(JSON.stringify({ results }, null, 2));
343
+ } else {
344
+ for (const result of results) {
345
+ printInstallSummary(result);
346
+ }
347
+ }
348
+ if (results.some((result) => result.conflicts.length)) {
349
+ process.exit(2);
350
+ }
351
+ } catch (error) {
352
+ if (parsed?.json) {
353
+ console.error(JSON.stringify({ error: error.message }, null, 2));
354
+ } else {
355
+ console.error(`[qingflow-skills] ${error.message}`);
356
+ console.error("Run 'qingflow-skills --help' for usage.");
357
+ }
358
+ process.exit(1);
359
+ }
69
360
  }
70
361
 
71
362
  export function getVenvDir(packageRoot) {
@@ -1,4 +1,4 @@
1
- import { ensurePythonEnv, getPackageRoot, installBundledSkills } from "../lib/runtime.mjs";
1
+ import { ensurePythonEnv, getPackageRoot } from "../lib/runtime.mjs";
2
2
 
3
3
  const packageRoot = getPackageRoot(import.meta.url);
4
4
 
@@ -6,10 +6,6 @@ try {
6
6
  console.log("[qingflow-mcp] Bootstrapping Python runtime...");
7
7
  ensurePythonEnv(packageRoot, { commandName: "qingflow-app-user-mcp" });
8
8
  console.log("[qingflow-mcp] Python runtime is ready.");
9
- const skills = installBundledSkills(packageRoot);
10
- if (!skills.skipped) {
11
- console.log(`[qingflow-mcp] Installed skills to ${skills.destination}: ${skills.installed.join(", ")}`);
12
- }
13
9
  } catch (error) {
14
10
  console.error(`[qingflow-mcp] postinstall failed: ${error.message}`);
15
11
  process.exit(1);
package/package.json CHANGED
@@ -1,11 +1,12 @@
1
1
  {
2
2
  "name": "@qingflow-tech/qingflow-app-user-mcp",
3
- "version": "1.0.11",
3
+ "version": "1.0.12",
4
4
  "description": "Operational end-user MCP for Qingflow records, tasks, comments, and directory workflows.",
5
5
  "license": "MIT",
6
6
  "type": "module",
7
7
  "bin": {
8
- "qingflow-app-user-mcp": "./npm/bin/qingflow-app-user-mcp.mjs"
8
+ "qingflow-app-user-mcp": "./npm/bin/qingflow-app-user-mcp.mjs",
9
+ "qingflow-app-user-mcp-skills": "./npm/bin/qingflow-skills.mjs"
9
10
  },
10
11
  "scripts": {
11
12
  "postinstall": "node ./npm/scripts/postinstall.mjs"
package/pyproject.toml CHANGED
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
4
4
 
5
5
  [project]
6
6
  name = "qingflow-mcp"
7
- version = "1.0.11"
7
+ version = "1.0.12"
8
8
  description = "User-authenticated MCP server for Qingflow"
9
9
  readme = "README.md"
10
10
  license = "MIT"