itismyskillmarket 1.2.2 → 1.2.4

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 (45) hide show
  1. package/.github/workflows/publish-npm.yml +54 -0
  2. package/.github/workflows/publish-skill.yml +72 -0
  3. package/5e51cb7aa8b8e60d49d86f4689f5d4d1.png +0 -0
  4. package/CHANGELOG.md +143 -0
  5. package/DEVELOPMENT.md +376 -0
  6. package/README.md +70 -4
  7. package/SKILLMARKET-GUIDE.md +277 -0
  8. package/dist/index.js +478 -177
  9. package/docs/plans/2026-04-01-skillmarket-design.md +267 -0
  10. package/docs/plans/2026-04-01-skillmarket-implementation.md +1031 -0
  11. package/docs/plans/2026-04-15-cross-platform-adapter-design.md +416 -0
  12. package/docs/plans/2026-04-15-cross-platform-adapter-plan.md +833 -0
  13. package/package.json +1 -6
  14. package/skills/README.md +52 -0
  15. package/skills/test-skill/SKILL.md +25 -0
  16. package/skills/test-skill/index.js +66 -0
  17. package/skills/test-skill/metadata.json +9 -0
  18. package/skills/test-skill/package.json +19 -0
  19. package/skills/test-skill-1/SKILL.md +24 -0
  20. package/skills/test-skill-1/index.js +13 -0
  21. package/skills/test-skill-1/metadata.json +9 -0
  22. package/skills/test-skill-1/package.json +16 -0
  23. package/skills/test-skill-2/SKILL.md +25 -0
  24. package/skills/test-skill-2/index.js +13 -0
  25. package/skills/test-skill-2/metadata.json +9 -0
  26. package/skills/test-skill-2/package.json +16 -0
  27. package/src/adapters/base.ts +87 -0
  28. package/src/adapters/claude.ts +31 -0
  29. package/src/adapters/index.ts +9 -0
  30. package/src/adapters/opencode.ts +40 -0
  31. package/src/adapters/registry.ts +77 -0
  32. package/src/adapters/vscode.ts +62 -0
  33. package/src/cli.ts +113 -49
  34. package/src/commands/info.ts +4 -15
  35. package/src/commands/install.ts +93 -54
  36. package/src/commands/ls.ts +69 -13
  37. package/src/commands/npm.ts +87 -12
  38. package/src/commands/sync.ts +6 -27
  39. package/src/commands/uninstall.ts +60 -7
  40. package/src/commands/update.ts +2 -2
  41. package/src/index.ts +27 -0
  42. package/src/types.ts +35 -0
  43. package/tsconfig.json +10 -0
  44. package/tsup.config.ts +22 -0
  45. package/wanxuchen-skillmarket-1.0.1.tgz +0 -0
package/src/cli.ts CHANGED
@@ -14,10 +14,11 @@
14
14
  * - skm ls 列出可用的 skills
15
15
  * - skm ls --installed 列出已安装的 skills
16
16
  * - skm info <skill> 显示 skill 详情
17
- * - skm install <skill> 安装 skill
18
- * - skm uninstall <skill> 卸载 skill
17
+ * - skm install <skill> 安装 skill(支持 --platform)
18
+ * - skm uninstall <skill> 卸载 skill(支持 --platform)
19
19
  * - skm update [skill] 更新 skill(s)
20
20
  * - skm sync 同步平台链接
21
+ * - skm platforms 显示可用平台
21
22
  *
22
23
  * @module cli
23
24
  */
@@ -37,6 +38,7 @@ import { installSkill } from './commands/install.js'; // 安装命令
37
38
  import { syncPlatformLinks } from './commands/sync.js'; // 同步命令
38
39
  import { updateSkill } from './commands/update.js'; // 更新命令
39
40
  import { uninstallSkill } from './commands/uninstall.js'; // 卸载命令
41
+ import { detectPlatforms, getAllAdapters, OpenCodeAdapter, ClaudeAdapter, VSCodeAdapter } from './adapters/index.js'; // 平台适配器
40
42
 
41
43
  // -----------------------------------------------------------------------------
42
44
  // 创建命令程序实例
@@ -63,7 +65,7 @@ const program = new Command();
63
65
  program
64
66
  .name('skm')
65
67
  .description('SkillMarket - Cross-platform skill manager for AI coding tools')
66
- .version('1.2.0');
68
+ .version('1.0.0');
67
69
 
68
70
  // -----------------------------------------------------------------------------
69
71
  // 帮助命令 (-h, --help)
@@ -74,38 +76,48 @@ program
74
76
  *
75
77
  * 显示详细的使用说明和命令示例
76
78
  */
77
- const helpCmd = program.command('help').description('Display help information');
78
- helpCmd.action(() => {
79
+ program
80
+ .hook('preAction', (thisCommand) => {
81
+ if (thisCommand.opts().help) {
79
82
  console.log(`
80
83
  SkillMarket CLI
81
84
 
82
85
  Usage: skm <command> [options]
83
86
 
84
87
  Commands:
85
- --help, -h Display this help message
86
- --ls [options] List available skills
87
- --installed Show only installed skills
88
- --updates Check for updates
89
- --info <skill-id> Display skill information
90
- --install <skill> Install a skill (e.g., skm --install brainstorming)
91
- @version Install specific version
92
- --all Install all available skills
93
- --uninstall <skill> Remove an installed skill
94
- --update [options] Update skills
95
- --all Update all skills
96
- --sync Synchronize platform links
97
- --platform <name> Set target platform (${PLATFORMS.join(', ')})
88
+ ls [options] List available skills
89
+ --installed Show only installed skills
90
+ --updates Check for updates
91
+ --page <n> Page number (default: 1)
92
+ --limit <n> Items per page (default: 20)
93
+ info <skill-id> Display skill information
94
+ install <skill> Install a skill
95
+ @version Install specific version
96
+ --platform Target platforms (opencode,claude,vscode)
97
+ --force Overwrite if already installed
98
+ uninstall <skill> Remove an installed skill
99
+ --platform Target platforms
100
+ update [options] Update skills
101
+ --all Update all skills
102
+ sync Synchronize platform links
103
+ platforms Show available platforms
98
104
 
99
105
  Examples:
100
- skm --ls List all available skills
101
- skm --ls --installed Show installed skills only
102
- skm --info brainstorming View skill details
103
- skm --install brainstorming Install a skill
104
- skm --install brainstorming@1.0.0 Install specific version
105
- skm --update --all Update all installed skills
106
- skm --sync Sync platform links
107
- `);
108
- });
106
+ skm ls List all available skills (page 1)
107
+ skm ls --page 2 Go to page 2
108
+ skm ls --limit 10 Show 10 items per page
109
+ skm ls --installed Show installed skills only
110
+ skm ls --installed --page 2
111
+ skm info brainstorming View skill details
112
+ skm install brainstorming Install to all platforms
113
+ skm install brainstorming --platform opencode Install to OpenCode only
114
+ skm install brainstorming --platform claude,vscode Install to multiple
115
+ skm uninstall brainstorming
116
+ skm platforms Show available platforms
117
+ `);
118
+ process.exit(0);
119
+ }
120
+ });
109
121
 
110
122
  // -----------------------------------------------------------------------------
111
123
  // 列表命令 (skm ls)
@@ -125,8 +137,16 @@ const lsCmd = program.command('ls').description('List available skills');
125
137
  lsCmd
126
138
  .option('--installed', 'Show only installed skills')
127
139
  .option('--updates', 'Check for updates')
140
+ .option('-p, --page <number>', 'Page number (default: 1)', parseInt)
141
+ .option('-l, --limit <number>', 'Items per page (default: 20)', parseInt)
128
142
  .action((opts) => {
129
- listSkills(opts);
143
+ // Ensure numeric options have default values if not provided
144
+ const options = {
145
+ ...opts,
146
+ page: opts.page ?? 1,
147
+ limit: opts.limit ?? 20
148
+ };
149
+ listSkills(options);
130
150
  });
131
151
 
132
152
  // -----------------------------------------------------------------------------
@@ -157,25 +177,36 @@ infoCmd
157
177
  /**
158
178
  * 安装命令
159
179
  *
160
- * 从 npm 安装指定的 skill 到本地
180
+ * 从 npm 安装指定的 skill 到本地和跨平台目录
161
181
  *
162
182
  * 用法:
163
- * - skm install <skill> 安装最新版本
183
+ * - skm install <skill> 安装到所有检测到的平台
164
184
  * - skm install <skill>@<ver> 安装指定版本
165
- * - skm install --all 安装所有可用 skills(预留)
185
+ * - skm install --platform opencode 安装到特定平台
186
+ * - skm install --platform claude,vscode 安装到多个平台
187
+ * - skm install --force 强制覆盖
166
188
  *
167
189
  * @example
168
190
  * skm install brainstorming
169
191
  * skm install brainstorming@1.0.0
192
+ * skm install brainstorming --platform opencode
170
193
  */
171
- const installCmd = program.command('install').description('Install a skill');
194
+ const installCmd = program.command('install').description('Install a skill to local and platform directories');
172
195
  installCmd
173
196
  .argument('<skill>', 'Skill ID to install (e.g., brainstorming or @scope/name)')
174
- .option('--all', 'Install all available skills')
175
- .option('-p, --platform <platform>', `Target platform (${PLATFORMS.join(', ')})`)
197
+ .option('-p, --platform <platforms>', 'Target platforms (comma-separated: opencode,claude,vscode)')
198
+ .option('-f, --force', 'Overwrite if already installed')
199
+ .option('-v, --version <version>', 'Specific version to install')
176
200
  .action(async (skill, opts) => {
177
201
  try {
178
- await installSkill(skill, undefined, opts.platform);
202
+ const platforms = opts.platform
203
+ ? opts.platform.split(',').map((p: string) => p.trim())
204
+ : undefined;
205
+
206
+ await installSkill(skill, opts.version, {
207
+ platforms,
208
+ force: opts.force
209
+ });
179
210
  } catch (err) {
180
211
  console.error('Installation failed:', err);
181
212
  process.exit(1);
@@ -189,19 +220,27 @@ installCmd
189
220
  /**
190
221
  * 卸载命令
191
222
  *
192
- * 移除本地已安装的 skill
223
+ * 移除本地已安装的 skill 及各平台的文件
193
224
  *
194
- * 用法: skm uninstall <skill-id>
225
+ * 用法:
226
+ * - skm uninstall <skill> 卸载所有平台
227
+ * - skm uninstall <skill> --platform opencode 卸载特定平台
195
228
  *
196
229
  * @example
197
230
  * skm uninstall brainstorming
231
+ * skm uninstall brainstorming --platform claude
198
232
  */
199
- const uninstallCmd = program.command('uninstall').description('Remove an installed skill');
233
+ const uninstallCmd = program.command('uninstall').description('Remove an installed skill from local and platform directories');
200
234
  uninstallCmd
201
235
  .argument('<skill>', 'Skill ID to uninstall')
202
- .action(async (skill) => {
236
+ .option('-p, --platform <platforms>', 'Target platforms (comma-separated)')
237
+ .action(async (skill, opts) => {
203
238
  try {
204
- await uninstallSkill(skill);
239
+ const platforms = opts.platform
240
+ ? opts.platform.split(',').map((p: string) => p.trim())
241
+ : undefined;
242
+
243
+ await uninstallSkill(skill, { platforms });
205
244
  } catch (err) {
206
245
  console.error('Uninstall failed:', err);
207
246
  process.exit(1);
@@ -268,21 +307,46 @@ program
268
307
  });
269
308
 
270
309
  // -----------------------------------------------------------------------------
271
- // 平台命令 (skm platform)
310
+ // 平台命令 (skm platforms)
272
311
  // -----------------------------------------------------------------------------
273
312
 
274
313
  /**
275
- * 平台命令(预留)
314
+ * 平台命令
276
315
  *
277
- * 设置默认目标平台
316
+ * 显示所有支持的平台及其状态
278
317
  *
279
- * 用法: skm platform <name>
318
+ * 用法: skm platforms
280
319
  */
281
- const platformCmd = program.command('platform').description('Set target platform');
282
- platformCmd
283
- .argument('<name>', 'Platform name')
284
- .action((name) => {
285
- console.log('Platform command - name:', name);
320
+ const platformsCmd = program.command('platforms').description('Show available platforms');
321
+ platformsCmd
322
+ .action(async () => {
323
+ try {
324
+ const available = await detectPlatforms();
325
+
326
+ console.log('\n📍 Available Platforms:\n');
327
+
328
+ const allPlatforms = [
329
+ { name: 'OpenCode', adapter: new OpenCodeAdapter() },
330
+ { name: 'Claude Code', adapter: new ClaudeAdapter() },
331
+ { name: 'VSCode', adapter: new VSCodeAdapter() },
332
+ ];
333
+
334
+ for (const { name, adapter } of allPlatforms) {
335
+ const isAvailable = available.find(a => a.id === adapter.id);
336
+ const installed = await adapter.listInstalled();
337
+
338
+ if (isAvailable) {
339
+ console.log(`${name.padEnd(12)} ✅ Available (${installed.length} skills installed)`);
340
+ } else {
341
+ console.log(`${name.padEnd(12)} ❌ Not detected`);
342
+ }
343
+ }
344
+
345
+ console.log('');
346
+ } catch (err) {
347
+ console.error('Failed to list platforms:', err);
348
+ process.exit(1);
349
+ }
286
350
  });
287
351
 
288
352
  // -----------------------------------------------------------------------------
@@ -19,7 +19,7 @@
19
19
  // 导入依赖
20
20
  // -----------------------------------------------------------------------------
21
21
 
22
- import { fetchNpmPackage } from './npm.js'; // npm registry 查询
22
+ import { fetchSkillPackage } from './npm.js'; // npm registry 查询
23
23
  import {
24
24
  isSkillInstalled, // 检查 skill 是否已安装
25
25
  getInstalledSkills // 获取已安装 skills 列表
@@ -49,25 +49,14 @@ export async function showSkillInfo(skillId: string): Promise<void> {
49
49
  // 处理包名格式
50
50
  // -------------------------------------------------------------------------
51
51
 
52
- /**
53
- * 转换 skillId 为完整的 npm 包名格式
54
- *
55
- * 支持两种输入格式:
56
- * 1. 短格式: "brainstorming" → "@skillmarket/brainstorming"
57
- * 2. 完整格式: "@custom/skill" → "@custom/skill"
58
- */
59
- const packageName = skillId.startsWith('@')
60
- ? skillId // 已经是 scoped 包名
61
- : `@skillmarket/${skillId}`; // 转换为 scoped 包名
62
-
63
- console.log(`Fetching info for: ${packageName}\n`);
52
+ console.log(`Fetching info for: ${skillId}\n`);
64
53
 
65
54
  try {
66
55
  // -------------------------------------------------------------------------
67
- // 从 npm 获取包信息
56
+ // 从 npm 获取包信息(自动尝试多个可能的 scope)
68
57
  // -------------------------------------------------------------------------
69
58
 
70
- const info = await fetchNpmPackage(packageName);
59
+ const info = await fetchSkillPackage(skillId);
71
60
 
72
61
  // 包不存在
73
62
  if (!info) {
@@ -3,7 +3,7 @@
3
3
  * SkillMarket 安装命令模块
4
4
  * =============================================================================
5
5
  *
6
- * 本模块实现 `skm install` 命令,用于安装 skill 到本地。
6
+ * 本模块实现 `skm install` 命令,用于安装 skill 到本地和跨平台目录。
7
7
  *
8
8
  * 安装流程:
9
9
  * 1. 确保目录结构存在
@@ -11,7 +11,8 @@
11
11
  * 3. 下载包到缓存
12
12
  * 4. 解压并复制到 skills 目录
13
13
  * 5. 创建 latest 软链接
14
- * 6. 更新本地注册表
14
+ * 6. 安装到目标平台(OpenCode/Claude Code/VSCode)
15
+ * 7. 更新本地注册表
15
16
  *
16
17
  * 安装后的目录结构:
17
18
  * ~/.skillmarket/
@@ -23,6 +24,11 @@
23
24
  * │ └── metadata.json
24
25
  * └── ...
25
26
  *
27
+ * 跨平台安装:
28
+ * - OpenCode: ~/.config/opencode/skills/<skillId>/SKILL.md
29
+ * - Claude Code: ~/.claude/skills/<skillId>/SKILL.md
30
+ * - VSCode: ~/.copilot/skills/<skillId>/SKILL.md
31
+ *
26
32
  * @module commands/install
27
33
  */
28
34
 
@@ -36,17 +42,29 @@ import { exec } from 'child_process'; // 执行 shell 命令
36
42
  import { promisify } from 'util'; // Promise 化工具
37
43
 
38
44
  // 模块导入
39
- import { fetchNpmPackage } from './npm.js'; // npm 查询
45
+ import { fetchSkillPackage } from './npm.js'; // npm 查询
40
46
  import { loadRegistry, saveRegistry } from './registry.js'; // 注册表操作
41
47
  import { getCacheDir, getSkillsDir, ensureMarketDirs } from '../utils/dirs.js'; // 目录工具
42
- import { detectPlatform, getPlatformFromInput, isValidPlatform, type Platform } from '../utils/platform.js'; // 平台检测
43
- import { syncPlatformLinks } from './sync.js'; // 同步命令
44
- import { LATEST_LINK, PLATFORMS } from '../constants.js'; // 常量
48
+ import { detectPlatforms, getAdapterByPlatform } from '../adapters/index.js'; // 平台适配器
49
+ import { LATEST_LINK } from '../constants.js'; // 常量
45
50
  import type { InstalledSkill } from '../types.js'; // 类型定义
51
+ import type { Platform } from '../constants.js';
52
+ import type { PlatformAdapter } from '../types.js';
46
53
 
47
54
  // 将 exec 转为 Promise 形式
48
55
  const execAsync = promisify(exec);
49
56
 
57
+ // -----------------------------------------------------------------------------
58
+ // 安装选项接口
59
+ // -----------------------------------------------------------------------------
60
+
61
+ export interface InstallOptions {
62
+ /** 目标平台列表(留空则安装到所有可用平台) */
63
+ platforms?: string[];
64
+ /** 强制覆盖已安装的 skill */
65
+ force?: boolean;
66
+ }
67
+
50
68
  // -----------------------------------------------------------------------------
51
69
  // 安装函数
52
70
  // -----------------------------------------------------------------------------
@@ -56,7 +74,7 @@ const execAsync = promisify(exec);
56
74
  *
57
75
  * @param {string} skillId - Skill 标识符(支持短格式或 scoped 格式)
58
76
  * @param {string} [version] - 指定版本号(可选,不指定则安装最新版本)
59
- * @param {string} [targetPlatform] - 目标平台(可选,不指定则使用当前检测到的平台)
77
+ * @param {InstallOptions} [options] - 安装选项
60
78
  * @returns {Promise<void>}
61
79
  *
62
80
  * @example
@@ -66,44 +84,39 @@ const execAsync = promisify(exec);
66
84
  * // 安装指定版本
67
85
  * await installSkill('brainstorming', '1.0.0');
68
86
  *
69
- * // 安装到指定平台
70
- * await installSkill('brainstorming', undefined, 'opencode');
87
+ * // 安装到特定平台
88
+ * await installSkill('brainstorming', undefined, { platforms: ['opencode'] });
71
89
  *
72
- * // 安装 scoped 包
73
- * await installSkill('@custom/skill');
90
+ * // 强制覆盖
91
+ * await installSkill('brainstorming', undefined, { force: true });
74
92
  */
75
93
  export async function installSkill(
76
94
  skillId: string,
77
95
  version?: string,
78
- targetPlatform?: string
96
+ options?: InstallOptions
79
97
  ): Promise<void> {
80
98
  // ==========================================================================
81
99
  // 步骤 0: 准备
82
100
  // ==========================================================================
83
101
 
84
- // 确保所有必要的目录都已创建
102
+ // 确保所有必要的目录都已创建
85
103
  await ensureMarketDirs();
86
-
87
- // 转换包名格式
88
- // 用户可以直接指定 @scope/package 或不带前缀的 short name
89
- // short name 会被添加 @skillmarket/ 前缀尝试
90
- // 如果是 scoped 包 (@scope/name),直接使用
91
- const isScoped = skillId.startsWith('@');
92
- const packageName = isScoped ? skillId : `@skillmarket/${skillId}`;
93
- const shortName = skillId; // 用于本地目录名
94
104
 
95
- console.log(`Installing ${packageName}${version ? `@${version}` : ''}...`);
105
+ console.log(`Installing ${skillId}${version ? `@${version}` : ''}...`);
96
106
 
97
107
  // ==========================================================================
98
108
  // 步骤 1: 获取包信息
99
109
  // ==========================================================================
100
110
 
101
- // 从 npm 查询包的元信息
102
- const pkgInfo = await fetchNpmPackage(packageName);
111
+ // 从 npm 查询包的元信息(自动尝试多个可能的 scope)
112
+ const pkgInfo = await fetchSkillPackage(skillId);
103
113
  if (!pkgInfo) {
104
- throw new Error(`Package ${packageName} not found`);
114
+ throw new Error(`Package ${skillId} not found`);
105
115
  }
106
116
 
117
+ // 获取实际找到的包名
118
+ const packageName = pkgInfo.name;
119
+
107
120
  // 确定要安装的版本(用户指定版本 > 最新版本)
108
121
  const targetVersion = version || pkgInfo['dist-tags']?.latest;
109
122
  if (!targetVersion) {
@@ -134,10 +147,10 @@ export async function installSkill(
134
147
  const files = await fs.readdir(cacheDir);
135
148
 
136
149
  // npm pack 生成的文件名格式: <package-name>-<version>.tgz
137
- // scoped 包格式: @scope-package-name-<version>.tgz
150
+ // scoped 包格式: @scope-package-name-<version>.tgz (注意:@ 不在文件名中)
138
151
  const tarball = files.find(f =>
139
152
  f.endsWith('.tgz') &&
140
- f.includes(packageName.replace('/', '-'))
153
+ f.includes(packageName.replace(/^@/, '').replace('/', '-'))
141
154
  );
142
155
 
143
156
  if (tarball) {
@@ -214,56 +227,82 @@ export async function installSkill(
214
227
  }
215
228
 
216
229
  // ==========================================================================
217
- // 步骤 5: 确定目标平台
230
+ // 步骤 5: 安装到目标平台 (NEW)
218
231
  // ==========================================================================
219
232
 
220
- // 确定实际使用的平台
221
- let finalPlatform: Platform;
222
-
223
- if (targetPlatform) {
224
- // 验证用户指定的目标平台
225
- const parsed = getPlatformFromInput(targetPlatform);
226
- if (!parsed) {
227
- throw new Error(`Invalid platform: ${targetPlatform}. Valid platforms: ${PLATFORMS.join(', ')}`);
233
+ let targetAdapters: PlatformAdapter[] = [];
234
+
235
+ if (options?.platforms && options.platforms.length > 0) {
236
+ // 用户指定了平台
237
+ for (const platformStr of options.platforms) {
238
+ const platform = platformStr as Platform;
239
+ const adapter = getAdapterByPlatform(platform);
240
+ if (adapter) {
241
+ targetAdapters.push(adapter);
242
+ } else {
243
+ console.warn(`⚠️ Unknown platform: ${platformStr}`);
244
+ }
228
245
  }
229
- finalPlatform = parsed;
230
246
  } else {
231
- // 使用当前检测到的平台
232
- finalPlatform = detectPlatform();
247
+ // 自动检测可用平台
248
+ targetAdapters = await detectPlatforms();
233
249
  }
234
250
 
235
- console.log(`Target platform: ${finalPlatform}`);
251
+ if (targetAdapters.length === 0) {
252
+ console.log('No target platforms detected.');
253
+ console.log('Use --platform to specify platforms manually.');
254
+ } else {
255
+ console.log(`\nInstalling to ${targetAdapters.length} platform(s)...\n`);
256
+
257
+ // 安装到每个平台
258
+ const results: { name: string; status: 'installed' | 'skipped' | 'failed'; error?: string }[] = [];
259
+
260
+ for (const adapter of targetAdapters) {
261
+ try {
262
+ const isInstalled = await adapter.isInstalled(skillId);
263
+
264
+ if (isInstalled && !options?.force) {
265
+ console.log(`${adapter.name.padEnd(12)} ⚠️ Already installed (use --force to overwrite)`);
266
+ results.push({ name: adapter.name, status: 'skipped' });
267
+ continue;
268
+ }
269
+
270
+ // 安装 skill 到平台目录
271
+ await adapter.install(skillId, skillVersionDir);
272
+ console.log(`${adapter.name.padEnd(12)} ✅ Installed successfully`);
273
+ results.push({ name: adapter.name, status: 'installed' });
274
+ } catch (error) {
275
+ console.log(`${adapter.name.padEnd(12)} ❌ Failed: ${error}`);
276
+ results.push({ name: adapter.name, status: 'failed', error: String(error) });
277
+ }
278
+ }
279
+
280
+ // 显示摘要
281
+ const installed = results.filter(r => r.status === 'installed').length;
282
+ const skipped = results.filter(r => r.status === 'skipped').length;
283
+ const failed = results.filter(r => r.status === 'failed').length;
284
+
285
+ console.log(`\n📊 Summary: ${installed} installed, ${skipped} skipped, ${failed} failed`);
286
+ }
236
287
 
237
288
  // ==========================================================================
238
289
  // 步骤 6: 更新注册表
239
290
  // ==========================================================================
240
291
 
241
292
  const registry = await loadRegistry();
242
-
243
- // 获取现有平台列表(如果 skill 已安装)
244
- const existingSkill = registry.skills[skillId];
245
- const existingPlatforms = existingSkill?.platforms || [];
293
+ const installedPlatforms = targetAdapters.map(a => a.id);
246
294
 
247
295
  // 添加/更新注册表中的 skill 记录
248
296
  registry.skills[skillId] = {
249
297
  id: skillId,
250
298
  version: targetVersion,
251
299
  installedAt: new Date().toISOString(),
252
- platforms: existingPlatforms.includes(finalPlatform as string)
253
- ? existingPlatforms
254
- : [...existingPlatforms, finalPlatform as string]
300
+ platforms: installedPlatforms
255
301
  } as InstalledSkill;
256
302
 
257
303
  // 保存注册表
258
304
  await saveRegistry(registry);
259
305
 
260
- // ==========================================================================
261
- // 步骤 7: 同步到指定平台
262
- // ==========================================================================
263
-
264
- console.log(`Syncing to ${finalPlatform}...`);
265
- await syncPlatformLinks(finalPlatform);
266
-
267
306
  // ==========================================================================
268
307
  // 完成
269
308
  // ==========================================================================