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.
- package/.github/workflows/publish-npm.yml +54 -0
- package/.github/workflows/publish-skill.yml +72 -0
- package/5e51cb7aa8b8e60d49d86f4689f5d4d1.png +0 -0
- package/CHANGELOG.md +143 -0
- package/DEVELOPMENT.md +376 -0
- package/README.md +70 -4
- package/SKILLMARKET-GUIDE.md +277 -0
- package/dist/index.js +478 -177
- package/docs/plans/2026-04-01-skillmarket-design.md +267 -0
- package/docs/plans/2026-04-01-skillmarket-implementation.md +1031 -0
- package/docs/plans/2026-04-15-cross-platform-adapter-design.md +416 -0
- package/docs/plans/2026-04-15-cross-platform-adapter-plan.md +833 -0
- package/package.json +1 -6
- package/skills/README.md +52 -0
- package/skills/test-skill/SKILL.md +25 -0
- package/skills/test-skill/index.js +66 -0
- package/skills/test-skill/metadata.json +9 -0
- package/skills/test-skill/package.json +19 -0
- package/skills/test-skill-1/SKILL.md +24 -0
- package/skills/test-skill-1/index.js +13 -0
- package/skills/test-skill-1/metadata.json +9 -0
- package/skills/test-skill-1/package.json +16 -0
- package/skills/test-skill-2/SKILL.md +25 -0
- package/skills/test-skill-2/index.js +13 -0
- package/skills/test-skill-2/metadata.json +9 -0
- package/skills/test-skill-2/package.json +16 -0
- package/src/adapters/base.ts +87 -0
- package/src/adapters/claude.ts +31 -0
- package/src/adapters/index.ts +9 -0
- package/src/adapters/opencode.ts +40 -0
- package/src/adapters/registry.ts +77 -0
- package/src/adapters/vscode.ts +62 -0
- package/src/cli.ts +113 -49
- package/src/commands/info.ts +4 -15
- package/src/commands/install.ts +93 -54
- package/src/commands/ls.ts +69 -13
- package/src/commands/npm.ts +87 -12
- package/src/commands/sync.ts +6 -27
- package/src/commands/uninstall.ts +60 -7
- package/src/commands/update.ts +2 -2
- package/src/index.ts +27 -0
- package/src/types.ts +35 -0
- package/tsconfig.json +10 -0
- package/tsup.config.ts +22 -0
- 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.
|
|
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
|
-
|
|
78
|
-
|
|
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
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
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
|
|
101
|
-
skm
|
|
102
|
-
skm --
|
|
103
|
-
skm --
|
|
104
|
-
skm --
|
|
105
|
-
skm
|
|
106
|
-
skm
|
|
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
|
-
|
|
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 --
|
|
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('--
|
|
175
|
-
.option('-
|
|
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
|
-
|
|
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
|
-
* 用法:
|
|
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
|
-
.
|
|
236
|
+
.option('-p, --platform <platforms>', 'Target platforms (comma-separated)')
|
|
237
|
+
.action(async (skill, opts) => {
|
|
203
238
|
try {
|
|
204
|
-
|
|
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
|
|
310
|
+
// 平台命令 (skm platforms)
|
|
272
311
|
// -----------------------------------------------------------------------------
|
|
273
312
|
|
|
274
313
|
/**
|
|
275
|
-
*
|
|
314
|
+
* 平台命令
|
|
276
315
|
*
|
|
277
|
-
*
|
|
316
|
+
* 显示所有支持的平台及其状态
|
|
278
317
|
*
|
|
279
|
-
* 用法: skm
|
|
318
|
+
* 用法: skm platforms
|
|
280
319
|
*/
|
|
281
|
-
const
|
|
282
|
-
|
|
283
|
-
.
|
|
284
|
-
|
|
285
|
-
|
|
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
|
// -----------------------------------------------------------------------------
|
package/src/commands/info.ts
CHANGED
|
@@ -19,7 +19,7 @@
|
|
|
19
19
|
// 导入依赖
|
|
20
20
|
// -----------------------------------------------------------------------------
|
|
21
21
|
|
|
22
|
-
import {
|
|
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
|
|
59
|
+
const info = await fetchSkillPackage(skillId);
|
|
71
60
|
|
|
72
61
|
// 包不存在
|
|
73
62
|
if (!info) {
|
package/src/commands/install.ts
CHANGED
|
@@ -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 {
|
|
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 {
|
|
43
|
-
import {
|
|
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 {
|
|
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
|
-
* //
|
|
73
|
-
* await installSkill('
|
|
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
|
-
|
|
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 ${
|
|
105
|
+
console.log(`Installing ${skillId}${version ? `@${version}` : ''}...`);
|
|
96
106
|
|
|
97
107
|
// ==========================================================================
|
|
98
108
|
// 步骤 1: 获取包信息
|
|
99
109
|
// ==========================================================================
|
|
100
110
|
|
|
101
|
-
// 从 npm
|
|
102
|
-
const pkgInfo = await
|
|
111
|
+
// 从 npm 查询包的元信息(自动尝试多个可能的 scope)
|
|
112
|
+
const pkgInfo = await fetchSkillPackage(skillId);
|
|
103
113
|
if (!pkgInfo) {
|
|
104
|
-
throw new Error(`Package ${
|
|
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
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
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
|
-
|
|
247
|
+
// 自动检测可用平台
|
|
248
|
+
targetAdapters = await detectPlatforms();
|
|
233
249
|
}
|
|
234
250
|
|
|
235
|
-
|
|
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:
|
|
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
|
// ==========================================================================
|