ccjk 2.0.7 → 2.0.9
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/README.ja.md +6 -6
- package/README.ko.md +5 -5
- package/README.md +55 -9
- package/README.zh-CN.md +9 -9
- package/dist/chunks/claude-code-config-manager.mjs +8 -8
- package/dist/chunks/claude-code-incremental-manager.mjs +2 -2
- package/dist/chunks/codex-config-switch.mjs +4 -4
- package/dist/chunks/codex-provider-manager.mjs +2 -2
- package/dist/chunks/codex-uninstaller.mjs +3 -3
- package/dist/chunks/commands.mjs +2 -2
- package/dist/chunks/features.mjs +11 -11
- package/dist/chunks/plugin-recommendation.mjs +555 -0
- package/dist/chunks/simple-config.mjs +231 -89
- package/dist/chunks/skills-sync.mjs +854 -0
- package/dist/cli.mjs +8187 -1317
- package/dist/i18n/locales/en/agents.json +135 -0
- package/dist/i18n/locales/en/claude-md.json +73 -0
- package/dist/i18n/locales/en/cloudPlugins.json +118 -0
- package/dist/i18n/locales/en/common.json +2 -1
- package/dist/i18n/locales/en/hooksSync.json +111 -0
- package/dist/i18n/locales/en/mcp.json +31 -6
- package/dist/i18n/locales/en/menu.json +60 -1
- package/dist/i18n/locales/en/notification.json +306 -0
- package/dist/i18n/locales/en/plugins.json +133 -0
- package/dist/i18n/locales/en/skillsSync.json +74 -0
- package/dist/i18n/locales/zh-CN/agents.json +135 -0
- package/dist/i18n/locales/zh-CN/claude-md.json +73 -0
- package/dist/i18n/locales/zh-CN/cloudPlugins.json +118 -0
- package/dist/i18n/locales/zh-CN/common.json +2 -1
- package/dist/i18n/locales/zh-CN/hooksSync.json +111 -0
- package/dist/i18n/locales/zh-CN/mcp.json +31 -6
- package/dist/i18n/locales/zh-CN/menu.json +60 -1
- package/dist/i18n/locales/zh-CN/notification.json +306 -0
- package/dist/i18n/locales/zh-CN/plugins.json +133 -0
- package/dist/i18n/locales/zh-CN/skillsSync.json +74 -0
- package/dist/index.d.mts +18 -18
- package/dist/index.d.ts +18 -18
- package/dist/index.mjs +190 -188
- package/package.json +52 -48
|
@@ -0,0 +1,854 @@
|
|
|
1
|
+
import ansis from 'ansis';
|
|
2
|
+
import inquirer from 'inquirer';
|
|
3
|
+
import { a as CCJK_SKILLS_DIR, P as CCJK_CONFIG_DIR, df as getTranslation } from './simple-config.mjs';
|
|
4
|
+
import { createHash } from 'node:crypto';
|
|
5
|
+
import { existsSync, readdirSync, readFileSync, mkdirSync, writeFileSync } from 'node:fs';
|
|
6
|
+
import process from 'node:process';
|
|
7
|
+
import { join } from 'pathe';
|
|
8
|
+
import { readFile, stat } from 'node:fs/promises';
|
|
9
|
+
import matter from 'gray-matter';
|
|
10
|
+
import 'node:os';
|
|
11
|
+
import 'smol-toml';
|
|
12
|
+
import 'dayjs';
|
|
13
|
+
import 'node:child_process';
|
|
14
|
+
import 'node:util';
|
|
15
|
+
import 'node:url';
|
|
16
|
+
import 'inquirer-toggle';
|
|
17
|
+
import 'ora';
|
|
18
|
+
import 'tinyexec';
|
|
19
|
+
import 'semver';
|
|
20
|
+
import 'fs-extra';
|
|
21
|
+
import 'trash';
|
|
22
|
+
import 'i18next';
|
|
23
|
+
import 'i18next-fs-backend';
|
|
24
|
+
|
|
25
|
+
function parseSkillMd(content, filePath = "unknown") {
|
|
26
|
+
try {
|
|
27
|
+
const parsed = matter(content);
|
|
28
|
+
const metadata = extractMetadata(parsed.data, filePath);
|
|
29
|
+
return {
|
|
30
|
+
metadata,
|
|
31
|
+
content: parsed.content.trim(),
|
|
32
|
+
filePath
|
|
33
|
+
};
|
|
34
|
+
} catch (error) {
|
|
35
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
36
|
+
throw new Error(`Failed to parse SKILL.md at ${filePath}: ${errorMessage}`);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
async function parseSkillMdFile(filePath) {
|
|
40
|
+
try {
|
|
41
|
+
const content = await readFile(filePath, "utf-8");
|
|
42
|
+
const skill = parseSkillMd(content, filePath);
|
|
43
|
+
try {
|
|
44
|
+
const stats = await stat(filePath);
|
|
45
|
+
skill.modifiedAt = stats.mtime;
|
|
46
|
+
} catch {
|
|
47
|
+
}
|
|
48
|
+
return skill;
|
|
49
|
+
} catch (error) {
|
|
50
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
51
|
+
throw new Error(`Failed to read SKILL.md file at ${filePath}: ${errorMessage}`);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
function extractMetadata(data, filePath) {
|
|
55
|
+
if (!data.name || typeof data.name !== "string") {
|
|
56
|
+
throw new Error(`Missing or invalid 'name' field in ${filePath}`);
|
|
57
|
+
}
|
|
58
|
+
if (!data.description || typeof data.description !== "string") {
|
|
59
|
+
throw new Error(`Missing or invalid 'description' field in ${filePath}`);
|
|
60
|
+
}
|
|
61
|
+
if (!data.version || typeof data.version !== "string") {
|
|
62
|
+
throw new Error(`Missing or invalid 'version' field in ${filePath}`);
|
|
63
|
+
}
|
|
64
|
+
if (!data.category || typeof data.category !== "string") {
|
|
65
|
+
throw new Error(`Missing or invalid 'category' field in ${filePath}`);
|
|
66
|
+
}
|
|
67
|
+
if (!Array.isArray(data.triggers) || data.triggers.length === 0) {
|
|
68
|
+
throw new Error(`Missing or invalid 'triggers' field in ${filePath}`);
|
|
69
|
+
}
|
|
70
|
+
if (!Array.isArray(data.use_when) || data.use_when.length === 0) {
|
|
71
|
+
throw new Error(`Missing or invalid 'use_when' field in ${filePath}`);
|
|
72
|
+
}
|
|
73
|
+
const metadata = {
|
|
74
|
+
name: data.name,
|
|
75
|
+
description: data.description,
|
|
76
|
+
version: data.version,
|
|
77
|
+
category: data.category,
|
|
78
|
+
triggers: data.triggers,
|
|
79
|
+
use_when: data.use_when
|
|
80
|
+
};
|
|
81
|
+
if (data.author && typeof data.author === "string") {
|
|
82
|
+
metadata.author = data.author;
|
|
83
|
+
}
|
|
84
|
+
if (typeof data.auto_activate === "boolean") {
|
|
85
|
+
metadata.auto_activate = data.auto_activate;
|
|
86
|
+
}
|
|
87
|
+
if (typeof data.priority === "number") {
|
|
88
|
+
metadata.priority = data.priority;
|
|
89
|
+
}
|
|
90
|
+
if (Array.isArray(data.agents)) {
|
|
91
|
+
metadata.agents = data.agents;
|
|
92
|
+
}
|
|
93
|
+
if (data.difficulty && typeof data.difficulty === "string") {
|
|
94
|
+
metadata.difficulty = data.difficulty;
|
|
95
|
+
}
|
|
96
|
+
if (Array.isArray(data.related_skills)) {
|
|
97
|
+
metadata.related_skills = data.related_skills;
|
|
98
|
+
}
|
|
99
|
+
if (data.ccjk_version && typeof data.ccjk_version === "string") {
|
|
100
|
+
metadata.ccjk_version = data.ccjk_version;
|
|
101
|
+
}
|
|
102
|
+
if (Array.isArray(data.tags)) {
|
|
103
|
+
metadata.tags = data.tags;
|
|
104
|
+
}
|
|
105
|
+
if (Array.isArray(data.allowed_tools)) {
|
|
106
|
+
metadata.allowed_tools = data.allowed_tools;
|
|
107
|
+
}
|
|
108
|
+
if (data.context && typeof data.context === "string") {
|
|
109
|
+
metadata.context = data.context;
|
|
110
|
+
}
|
|
111
|
+
if (data.agent && typeof data.agent === "string") {
|
|
112
|
+
metadata.agent = data.agent;
|
|
113
|
+
}
|
|
114
|
+
if (typeof data.user_invocable === "boolean") {
|
|
115
|
+
metadata.user_invocable = data.user_invocable;
|
|
116
|
+
}
|
|
117
|
+
if (Array.isArray(data.hooks)) {
|
|
118
|
+
metadata.hooks = data.hooks;
|
|
119
|
+
}
|
|
120
|
+
if (Array.isArray(data.permissions)) {
|
|
121
|
+
metadata.permissions = data.permissions;
|
|
122
|
+
}
|
|
123
|
+
if (typeof data.timeout === "number") {
|
|
124
|
+
metadata.timeout = data.timeout;
|
|
125
|
+
}
|
|
126
|
+
if (Array.isArray(data.outputs)) {
|
|
127
|
+
metadata.outputs = data.outputs;
|
|
128
|
+
}
|
|
129
|
+
return metadata;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
const CLOUD_API_BASE_URL = process.env.CCJK_CLOUD_API_URL || "https://api.ccjk.dev/v1";
|
|
133
|
+
const SYNC_STATE_FILE = join(CCJK_CONFIG_DIR, "skills-sync-state.json");
|
|
134
|
+
const DEFAULT_TIMEOUT = 3e4;
|
|
135
|
+
function loadSyncState() {
|
|
136
|
+
if (!existsSync(SYNC_STATE_FILE)) {
|
|
137
|
+
return {
|
|
138
|
+
version: "1.0.0",
|
|
139
|
+
lastGlobalSync: (/* @__PURE__ */ new Date()).toISOString(),
|
|
140
|
+
skills: {}
|
|
141
|
+
};
|
|
142
|
+
}
|
|
143
|
+
try {
|
|
144
|
+
const content = readFileSync(SYNC_STATE_FILE, "utf-8");
|
|
145
|
+
return JSON.parse(content);
|
|
146
|
+
} catch (error) {
|
|
147
|
+
console.warn("Failed to load sync state, using empty state:", error);
|
|
148
|
+
return {
|
|
149
|
+
version: "1.0.0",
|
|
150
|
+
lastGlobalSync: (/* @__PURE__ */ new Date()).toISOString(),
|
|
151
|
+
skills: {}
|
|
152
|
+
};
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
function saveSyncState(state) {
|
|
156
|
+
try {
|
|
157
|
+
if (!existsSync(CCJK_CONFIG_DIR)) {
|
|
158
|
+
mkdirSync(CCJK_CONFIG_DIR, { recursive: true });
|
|
159
|
+
}
|
|
160
|
+
writeFileSync(SYNC_STATE_FILE, JSON.stringify(state, null, 2), "utf-8");
|
|
161
|
+
} catch (error) {
|
|
162
|
+
console.error("Failed to save sync state:", error);
|
|
163
|
+
throw new Error(`Failed to save sync state: ${error}`);
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
function updateSyncState(skillId, localVersion, remoteVersion, localChecksum, remoteChecksum) {
|
|
167
|
+
const state = loadSyncState();
|
|
168
|
+
let status = "synced";
|
|
169
|
+
if (localChecksum === remoteChecksum) {
|
|
170
|
+
status = "synced";
|
|
171
|
+
} else if (!remoteChecksum) {
|
|
172
|
+
status = "local_only";
|
|
173
|
+
} else if (!localChecksum) {
|
|
174
|
+
status = "remote_only";
|
|
175
|
+
} else if (localVersion > remoteVersion) {
|
|
176
|
+
status = "local_ahead";
|
|
177
|
+
} else if (remoteVersion > localVersion) {
|
|
178
|
+
status = "remote_ahead";
|
|
179
|
+
} else {
|
|
180
|
+
status = "conflict";
|
|
181
|
+
}
|
|
182
|
+
state.skills[skillId] = {
|
|
183
|
+
skillId,
|
|
184
|
+
lastSyncTime: (/* @__PURE__ */ new Date()).toISOString(),
|
|
185
|
+
localVersion,
|
|
186
|
+
remoteVersion,
|
|
187
|
+
localChecksum,
|
|
188
|
+
remoteChecksum,
|
|
189
|
+
status
|
|
190
|
+
};
|
|
191
|
+
state.lastGlobalSync = (/* @__PURE__ */ new Date()).toISOString();
|
|
192
|
+
saveSyncState(state);
|
|
193
|
+
}
|
|
194
|
+
function calculateChecksum(content) {
|
|
195
|
+
return createHash("sha256").update(content).digest("hex");
|
|
196
|
+
}
|
|
197
|
+
async function getLocalSkills() {
|
|
198
|
+
if (!existsSync(CCJK_SKILLS_DIR)) {
|
|
199
|
+
return [];
|
|
200
|
+
}
|
|
201
|
+
const skills = [];
|
|
202
|
+
const files = readdirSync(CCJK_SKILLS_DIR);
|
|
203
|
+
for (const file of files) {
|
|
204
|
+
if (file.endsWith(".md")) {
|
|
205
|
+
const filePath = join(CCJK_SKILLS_DIR, file);
|
|
206
|
+
try {
|
|
207
|
+
const skill = await parseSkillMdFile(filePath);
|
|
208
|
+
skills.push(skill);
|
|
209
|
+
} catch (error) {
|
|
210
|
+
console.warn(`Failed to parse skill file ${file}:`, error);
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
return skills;
|
|
215
|
+
}
|
|
216
|
+
async function getLocalSkill(skillId) {
|
|
217
|
+
const filePath = join(CCJK_SKILLS_DIR, `${skillId}.md`);
|
|
218
|
+
if (!existsSync(filePath)) {
|
|
219
|
+
return null;
|
|
220
|
+
}
|
|
221
|
+
try {
|
|
222
|
+
return await parseSkillMdFile(filePath);
|
|
223
|
+
} catch (error) {
|
|
224
|
+
console.error(`Failed to parse skill ${skillId}:`, error);
|
|
225
|
+
return null;
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
function saveLocalSkill(skillId, content) {
|
|
229
|
+
if (!existsSync(CCJK_SKILLS_DIR)) {
|
|
230
|
+
mkdirSync(CCJK_SKILLS_DIR, { recursive: true });
|
|
231
|
+
}
|
|
232
|
+
const filePath = join(CCJK_SKILLS_DIR, `${skillId}.md`);
|
|
233
|
+
writeFileSync(filePath, content, "utf-8");
|
|
234
|
+
}
|
|
235
|
+
function getAuthToken() {
|
|
236
|
+
const tokenFile = join(CCJK_CONFIG_DIR, "cloud-token.json");
|
|
237
|
+
if (!existsSync(tokenFile)) {
|
|
238
|
+
return null;
|
|
239
|
+
}
|
|
240
|
+
try {
|
|
241
|
+
const content = readFileSync(tokenFile, "utf-8");
|
|
242
|
+
const data = JSON.parse(content);
|
|
243
|
+
return data.deviceToken || null;
|
|
244
|
+
} catch {
|
|
245
|
+
return null;
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
async function apiRequest(endpoint, options = {}) {
|
|
249
|
+
const token = getAuthToken();
|
|
250
|
+
if (!token) {
|
|
251
|
+
throw new Error("Not authenticated. Please bind device first using: npx ccjk notification bind");
|
|
252
|
+
}
|
|
253
|
+
const url = `${CLOUD_API_BASE_URL}${endpoint}`;
|
|
254
|
+
const headers = {
|
|
255
|
+
"Content-Type": "application/json",
|
|
256
|
+
"Authorization": `Bearer ${token}`,
|
|
257
|
+
...options.headers
|
|
258
|
+
};
|
|
259
|
+
try {
|
|
260
|
+
const controller = new AbortController();
|
|
261
|
+
const timeoutId = setTimeout(() => controller.abort(), DEFAULT_TIMEOUT);
|
|
262
|
+
const response = await fetch(url, {
|
|
263
|
+
...options,
|
|
264
|
+
headers,
|
|
265
|
+
signal: controller.signal
|
|
266
|
+
});
|
|
267
|
+
clearTimeout(timeoutId);
|
|
268
|
+
const data = await response.json();
|
|
269
|
+
if (!response.ok) {
|
|
270
|
+
return {
|
|
271
|
+
success: false,
|
|
272
|
+
error: data.error || `HTTP ${response.status}: ${response.statusText}`,
|
|
273
|
+
code: data.code || String(response.status),
|
|
274
|
+
timestamp: (/* @__PURE__ */ new Date()).toISOString()
|
|
275
|
+
};
|
|
276
|
+
}
|
|
277
|
+
return {
|
|
278
|
+
success: true,
|
|
279
|
+
data,
|
|
280
|
+
timestamp: (/* @__PURE__ */ new Date()).toISOString()
|
|
281
|
+
};
|
|
282
|
+
} catch (error) {
|
|
283
|
+
return {
|
|
284
|
+
success: false,
|
|
285
|
+
error: error instanceof Error ? error.message : String(error),
|
|
286
|
+
timestamp: (/* @__PURE__ */ new Date()).toISOString()
|
|
287
|
+
};
|
|
288
|
+
}
|
|
289
|
+
}
|
|
290
|
+
async function listCloudSkills(options = {}) {
|
|
291
|
+
const params = new URLSearchParams();
|
|
292
|
+
if (options.privacy)
|
|
293
|
+
params.append("privacy", options.privacy);
|
|
294
|
+
if (options.author)
|
|
295
|
+
params.append("author", options.author);
|
|
296
|
+
if (options.tags)
|
|
297
|
+
params.append("tags", options.tags.join(","));
|
|
298
|
+
if (options.query)
|
|
299
|
+
params.append("query", options.query);
|
|
300
|
+
if (options.page)
|
|
301
|
+
params.append("page", String(options.page));
|
|
302
|
+
if (options.pageSize)
|
|
303
|
+
params.append("pageSize", String(options.pageSize));
|
|
304
|
+
if (options.sortBy)
|
|
305
|
+
params.append("sortBy", options.sortBy);
|
|
306
|
+
if (options.sortDir)
|
|
307
|
+
params.append("sortDir", options.sortDir);
|
|
308
|
+
const queryString = params.toString();
|
|
309
|
+
const endpoint = `/skills${queryString ? `?${queryString}` : ""}`;
|
|
310
|
+
return apiRequest(endpoint, { method: "GET" });
|
|
311
|
+
}
|
|
312
|
+
async function getCloudSkill(skillId, version) {
|
|
313
|
+
const endpoint = `/skills/${skillId}`;
|
|
314
|
+
return apiRequest(endpoint, { method: "GET" });
|
|
315
|
+
}
|
|
316
|
+
async function uploadSkill(request) {
|
|
317
|
+
return apiRequest("/skills", {
|
|
318
|
+
method: "POST",
|
|
319
|
+
body: JSON.stringify(request)
|
|
320
|
+
});
|
|
321
|
+
}
|
|
322
|
+
async function updateCloudSkill(skillId, request) {
|
|
323
|
+
return apiRequest(`/skills/${skillId}`, {
|
|
324
|
+
method: "PUT",
|
|
325
|
+
body: JSON.stringify(request)
|
|
326
|
+
});
|
|
327
|
+
}
|
|
328
|
+
async function syncSkill(skillId, options = {}) {
|
|
329
|
+
try {
|
|
330
|
+
const localSkill = await getLocalSkill(skillId);
|
|
331
|
+
const localContent = localSkill ? readFileSync(join(CCJK_SKILLS_DIR, `${skillId}.md`), "utf-8") : "";
|
|
332
|
+
const localChecksum = localContent ? calculateChecksum(localContent) : "";
|
|
333
|
+
const localVersion = localSkill?.metadata.version || "";
|
|
334
|
+
const cloudResponse = await getCloudSkill(skillId);
|
|
335
|
+
const cloudSkill = cloudResponse.success ? cloudResponse.data : null;
|
|
336
|
+
const remoteChecksum = cloudSkill?.checksum || "";
|
|
337
|
+
const remoteVersion = cloudSkill?.version || "";
|
|
338
|
+
const syncState = loadSyncState();
|
|
339
|
+
const previousState = syncState.skills[skillId];
|
|
340
|
+
let action = "skipped";
|
|
341
|
+
let newState;
|
|
342
|
+
if (localSkill && cloudSkill) {
|
|
343
|
+
if (localChecksum === remoteChecksum) {
|
|
344
|
+
action = "skipped";
|
|
345
|
+
} else if (options.force) {
|
|
346
|
+
if (!options.dryRun) {
|
|
347
|
+
const uploadRequest = {
|
|
348
|
+
name: localSkill.metadata.name,
|
|
349
|
+
version: localSkill.metadata.version,
|
|
350
|
+
content: localContent,
|
|
351
|
+
metadata: {
|
|
352
|
+
author: localSkill.metadata.author || "unknown",
|
|
353
|
+
description: localSkill.metadata.description,
|
|
354
|
+
tags: localSkill.metadata.tags || [],
|
|
355
|
+
category: localSkill.metadata.category
|
|
356
|
+
},
|
|
357
|
+
privacy: "private",
|
|
358
|
+
checksum: localChecksum
|
|
359
|
+
};
|
|
360
|
+
await updateCloudSkill(skillId, uploadRequest);
|
|
361
|
+
}
|
|
362
|
+
action = "uploaded";
|
|
363
|
+
} else {
|
|
364
|
+
const resolution = options.conflictResolution || "prompt";
|
|
365
|
+
if (resolution === "local") {
|
|
366
|
+
if (!options.dryRun) {
|
|
367
|
+
const uploadRequest = {
|
|
368
|
+
name: localSkill.metadata.name,
|
|
369
|
+
version: localSkill.metadata.version,
|
|
370
|
+
content: localContent,
|
|
371
|
+
metadata: {
|
|
372
|
+
author: localSkill.metadata.author || "unknown",
|
|
373
|
+
description: localSkill.metadata.description,
|
|
374
|
+
tags: localSkill.metadata.tags || [],
|
|
375
|
+
category: localSkill.metadata.category
|
|
376
|
+
},
|
|
377
|
+
privacy: "private",
|
|
378
|
+
checksum: localChecksum
|
|
379
|
+
};
|
|
380
|
+
await updateCloudSkill(skillId, uploadRequest);
|
|
381
|
+
}
|
|
382
|
+
action = "uploaded";
|
|
383
|
+
} else if (resolution === "remote") {
|
|
384
|
+
if (!options.dryRun) {
|
|
385
|
+
saveLocalSkill(skillId, cloudSkill.content);
|
|
386
|
+
}
|
|
387
|
+
action = "downloaded";
|
|
388
|
+
} else if (resolution === "newer") {
|
|
389
|
+
const localTime = localSkill.modifiedAt?.getTime() || 0;
|
|
390
|
+
const remoteTime = new Date(cloudSkill.updatedAt).getTime();
|
|
391
|
+
if (localTime > remoteTime) {
|
|
392
|
+
if (!options.dryRun) {
|
|
393
|
+
const uploadRequest = {
|
|
394
|
+
name: localSkill.metadata.name,
|
|
395
|
+
version: localSkill.metadata.version,
|
|
396
|
+
content: localContent,
|
|
397
|
+
metadata: {
|
|
398
|
+
author: localSkill.metadata.author || "unknown",
|
|
399
|
+
description: localSkill.metadata.description,
|
|
400
|
+
tags: localSkill.metadata.tags || [],
|
|
401
|
+
category: localSkill.metadata.category
|
|
402
|
+
},
|
|
403
|
+
privacy: "private",
|
|
404
|
+
checksum: localChecksum
|
|
405
|
+
};
|
|
406
|
+
await updateCloudSkill(skillId, uploadRequest);
|
|
407
|
+
}
|
|
408
|
+
action = "uploaded";
|
|
409
|
+
} else {
|
|
410
|
+
if (!options.dryRun) {
|
|
411
|
+
saveLocalSkill(skillId, cloudSkill.content);
|
|
412
|
+
}
|
|
413
|
+
action = "downloaded";
|
|
414
|
+
}
|
|
415
|
+
} else {
|
|
416
|
+
action = "conflict";
|
|
417
|
+
}
|
|
418
|
+
}
|
|
419
|
+
} else if (localSkill && !cloudSkill) {
|
|
420
|
+
if (!options.dryRun) {
|
|
421
|
+
const uploadRequest = {
|
|
422
|
+
name: localSkill.metadata.name,
|
|
423
|
+
version: localSkill.metadata.version,
|
|
424
|
+
content: localContent,
|
|
425
|
+
metadata: {
|
|
426
|
+
author: localSkill.metadata.author || "unknown",
|
|
427
|
+
description: localSkill.metadata.description,
|
|
428
|
+
tags: localSkill.metadata.tags || [],
|
|
429
|
+
category: localSkill.metadata.category
|
|
430
|
+
},
|
|
431
|
+
privacy: "private",
|
|
432
|
+
checksum: localChecksum
|
|
433
|
+
};
|
|
434
|
+
await uploadSkill(uploadRequest);
|
|
435
|
+
}
|
|
436
|
+
action = "uploaded";
|
|
437
|
+
} else if (!localSkill && cloudSkill) {
|
|
438
|
+
if (!options.dryRun) {
|
|
439
|
+
saveLocalSkill(skillId, cloudSkill.content);
|
|
440
|
+
}
|
|
441
|
+
action = "downloaded";
|
|
442
|
+
}
|
|
443
|
+
if (!options.dryRun && action !== "conflict") {
|
|
444
|
+
updateSyncState(skillId, localVersion, remoteVersion, localChecksum, remoteChecksum);
|
|
445
|
+
newState = loadSyncState().skills[skillId];
|
|
446
|
+
}
|
|
447
|
+
return {
|
|
448
|
+
skillId,
|
|
449
|
+
skillName: localSkill?.metadata.name || cloudSkill?.name || skillId,
|
|
450
|
+
success: true,
|
|
451
|
+
action,
|
|
452
|
+
previousState,
|
|
453
|
+
newState
|
|
454
|
+
};
|
|
455
|
+
} catch (error) {
|
|
456
|
+
return {
|
|
457
|
+
skillId,
|
|
458
|
+
skillName: skillId,
|
|
459
|
+
success: false,
|
|
460
|
+
action: "skipped",
|
|
461
|
+
error: error instanceof Error ? error.message : String(error)
|
|
462
|
+
};
|
|
463
|
+
}
|
|
464
|
+
}
|
|
465
|
+
async function syncAllSkills(options = {}) {
|
|
466
|
+
const startTime = Date.now();
|
|
467
|
+
try {
|
|
468
|
+
const localSkills = await getLocalSkills();
|
|
469
|
+
const localSkillIds = new Set(localSkills.map((s) => s.metadata.name));
|
|
470
|
+
const cloudResponse = await listCloudSkills({ privacy: options.privacy });
|
|
471
|
+
if (!cloudResponse.success) {
|
|
472
|
+
throw new Error(cloudResponse.error || "Failed to list cloud skills");
|
|
473
|
+
}
|
|
474
|
+
const cloudSkills = cloudResponse.data?.skills || [];
|
|
475
|
+
const cloudSkillIds = new Set(cloudSkills.map((s) => s.id));
|
|
476
|
+
const allSkillIds = /* @__PURE__ */ new Set([...localSkillIds, ...cloudSkillIds]);
|
|
477
|
+
let skillIdsToSync = Array.from(allSkillIds);
|
|
478
|
+
if (options.skillIds && options.skillIds.length > 0) {
|
|
479
|
+
skillIdsToSync = skillIdsToSync.filter((id) => options.skillIds.includes(id));
|
|
480
|
+
}
|
|
481
|
+
const results = [];
|
|
482
|
+
for (const skillId of skillIdsToSync) {
|
|
483
|
+
const result = await syncSkill(skillId, options);
|
|
484
|
+
results.push(result);
|
|
485
|
+
}
|
|
486
|
+
const succeeded = results.filter((r) => r.success).length;
|
|
487
|
+
const failed = results.filter((r) => !r.success).length;
|
|
488
|
+
const conflicts = results.filter((r) => r.action === "conflict").length;
|
|
489
|
+
const uploaded = results.filter((r) => r.action === "uploaded").length;
|
|
490
|
+
const downloaded = results.filter((r) => r.action === "downloaded").length;
|
|
491
|
+
const skipped = results.filter((r) => r.action === "skipped").length;
|
|
492
|
+
return {
|
|
493
|
+
success: failed === 0,
|
|
494
|
+
total: results.length,
|
|
495
|
+
succeeded,
|
|
496
|
+
failed,
|
|
497
|
+
conflicts,
|
|
498
|
+
uploaded,
|
|
499
|
+
downloaded,
|
|
500
|
+
skipped,
|
|
501
|
+
results,
|
|
502
|
+
durationMs: Date.now() - startTime
|
|
503
|
+
};
|
|
504
|
+
} catch (error) {
|
|
505
|
+
return {
|
|
506
|
+
success: false,
|
|
507
|
+
total: 0,
|
|
508
|
+
succeeded: 0,
|
|
509
|
+
failed: 0,
|
|
510
|
+
conflicts: 0,
|
|
511
|
+
uploaded: 0,
|
|
512
|
+
downloaded: 0,
|
|
513
|
+
skipped: 0,
|
|
514
|
+
results: [],
|
|
515
|
+
error: error instanceof Error ? error.message : String(error),
|
|
516
|
+
durationMs: Date.now() - startTime
|
|
517
|
+
};
|
|
518
|
+
}
|
|
519
|
+
}
|
|
520
|
+
async function pushSkills(skillIds, options = {}) {
|
|
521
|
+
return syncAllSkills({
|
|
522
|
+
...options,
|
|
523
|
+
skillIds,
|
|
524
|
+
conflictResolution: options.conflictResolution || "local"
|
|
525
|
+
});
|
|
526
|
+
}
|
|
527
|
+
async function pullSkills(skillIds, options = {}) {
|
|
528
|
+
return syncAllSkills({
|
|
529
|
+
...options,
|
|
530
|
+
skillIds,
|
|
531
|
+
conflictResolution: options.conflictResolution || "remote"
|
|
532
|
+
});
|
|
533
|
+
}
|
|
534
|
+
|
|
535
|
+
async function syncSkills(options = {}) {
|
|
536
|
+
const t = getTranslation(options.lang);
|
|
537
|
+
console.log("");
|
|
538
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
539
|
+
console.log(ansis.bold.cyan(` ${t("skillsSync:title.sync")}`));
|
|
540
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
541
|
+
console.log("");
|
|
542
|
+
try {
|
|
543
|
+
await showSyncStatus(options);
|
|
544
|
+
const { confirm } = await inquirer.prompt({
|
|
545
|
+
type: "confirm",
|
|
546
|
+
name: "confirm",
|
|
547
|
+
message: t("skillsSync:prompt.confirmSync"),
|
|
548
|
+
default: true
|
|
549
|
+
});
|
|
550
|
+
if (!confirm) {
|
|
551
|
+
console.log(ansis.yellow(`
|
|
552
|
+
${t("skillsSync:message.cancelled")}`));
|
|
553
|
+
return;
|
|
554
|
+
}
|
|
555
|
+
console.log(ansis.dim(`
|
|
556
|
+
${t("skillsSync:message.syncing")}...
|
|
557
|
+
`));
|
|
558
|
+
const syncOptions = {
|
|
559
|
+
conflictResolution: options.conflictResolution || "prompt",
|
|
560
|
+
dryRun: options.dryRun,
|
|
561
|
+
force: options.force,
|
|
562
|
+
skillIds: options.skillIds,
|
|
563
|
+
privacy: options.privacy
|
|
564
|
+
};
|
|
565
|
+
const result = await syncAllSkills(syncOptions);
|
|
566
|
+
displaySyncResult(result, options.lang);
|
|
567
|
+
} catch (error) {
|
|
568
|
+
console.error(ansis.red(`
|
|
569
|
+
${t("skillsSync:error.syncFailed")}: ${error}`));
|
|
570
|
+
throw error;
|
|
571
|
+
}
|
|
572
|
+
}
|
|
573
|
+
async function pushSkillsCommand(options = {}) {
|
|
574
|
+
const t = getTranslation(options.lang);
|
|
575
|
+
console.log("");
|
|
576
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
577
|
+
console.log(ansis.bold.cyan(` ${t("skillsSync:title.push")}`));
|
|
578
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
579
|
+
console.log("");
|
|
580
|
+
try {
|
|
581
|
+
const localSkills = await getLocalSkills();
|
|
582
|
+
console.log(ansis.dim(` ${t("skillsSync:message.foundLocalSkills", { count: localSkills.length })}`));
|
|
583
|
+
if (localSkills.length === 0) {
|
|
584
|
+
console.log(ansis.yellow(`
|
|
585
|
+
${t("skillsSync:message.noLocalSkills")}`));
|
|
586
|
+
return;
|
|
587
|
+
}
|
|
588
|
+
const { confirm } = await inquirer.prompt({
|
|
589
|
+
type: "confirm",
|
|
590
|
+
name: "confirm",
|
|
591
|
+
message: t("skillsSync:prompt.confirmPush"),
|
|
592
|
+
default: true
|
|
593
|
+
});
|
|
594
|
+
if (!confirm) {
|
|
595
|
+
console.log(ansis.yellow(`
|
|
596
|
+
${t("skillsSync:message.cancelled")}`));
|
|
597
|
+
return;
|
|
598
|
+
}
|
|
599
|
+
console.log(ansis.dim(`
|
|
600
|
+
${t("skillsSync:message.pushing")}...
|
|
601
|
+
`));
|
|
602
|
+
const syncOptions = {
|
|
603
|
+
conflictResolution: "local",
|
|
604
|
+
dryRun: options.dryRun,
|
|
605
|
+
force: options.force,
|
|
606
|
+
skillIds: options.skillIds,
|
|
607
|
+
privacy: options.privacy
|
|
608
|
+
};
|
|
609
|
+
const result = await pushSkills(options.skillIds, syncOptions);
|
|
610
|
+
displaySyncResult(result, options.lang);
|
|
611
|
+
} catch (error) {
|
|
612
|
+
console.error(ansis.red(`
|
|
613
|
+
${t("skillsSync:error.pushFailed")}: ${error}`));
|
|
614
|
+
throw error;
|
|
615
|
+
}
|
|
616
|
+
}
|
|
617
|
+
async function pullSkillsCommand(options = {}) {
|
|
618
|
+
const t = getTranslation(options.lang);
|
|
619
|
+
console.log("");
|
|
620
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
621
|
+
console.log(ansis.bold.cyan(` ${t("skillsSync:title.pull")}`));
|
|
622
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
623
|
+
console.log("");
|
|
624
|
+
try {
|
|
625
|
+
const cloudResponse = await listCloudSkills({ privacy: options.privacy });
|
|
626
|
+
if (!cloudResponse.success) {
|
|
627
|
+
throw new Error(cloudResponse.error || "Failed to list cloud skills");
|
|
628
|
+
}
|
|
629
|
+
const cloudSkills = cloudResponse.data?.skills || [];
|
|
630
|
+
console.log(ansis.dim(` ${t("skillsSync:message.foundCloudSkills", { count: cloudSkills.length })}`));
|
|
631
|
+
if (cloudSkills.length === 0) {
|
|
632
|
+
console.log(ansis.yellow(`
|
|
633
|
+
${t("skillsSync:message.noCloudSkills")}`));
|
|
634
|
+
return;
|
|
635
|
+
}
|
|
636
|
+
const { confirm } = await inquirer.prompt({
|
|
637
|
+
type: "confirm",
|
|
638
|
+
name: "confirm",
|
|
639
|
+
message: t("skillsSync:prompt.confirmPull"),
|
|
640
|
+
default: true
|
|
641
|
+
});
|
|
642
|
+
if (!confirm) {
|
|
643
|
+
console.log(ansis.yellow(`
|
|
644
|
+
${t("skillsSync:message.cancelled")}`));
|
|
645
|
+
return;
|
|
646
|
+
}
|
|
647
|
+
console.log(ansis.dim(`
|
|
648
|
+
${t("skillsSync:message.pulling")}...
|
|
649
|
+
`));
|
|
650
|
+
const syncOptions = {
|
|
651
|
+
conflictResolution: "remote",
|
|
652
|
+
dryRun: options.dryRun,
|
|
653
|
+
force: options.force,
|
|
654
|
+
skillIds: options.skillIds,
|
|
655
|
+
privacy: options.privacy
|
|
656
|
+
};
|
|
657
|
+
const result = await pullSkills(options.skillIds, syncOptions);
|
|
658
|
+
displaySyncResult(result, options.lang);
|
|
659
|
+
} catch (error) {
|
|
660
|
+
console.error(ansis.red(`
|
|
661
|
+
${t("skillsSync:error.pullFailed")}: ${error}`));
|
|
662
|
+
throw error;
|
|
663
|
+
}
|
|
664
|
+
}
|
|
665
|
+
async function listCloudSkillsCommand(options = {}) {
|
|
666
|
+
const t = getTranslation(options.lang);
|
|
667
|
+
console.log("");
|
|
668
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
669
|
+
console.log(ansis.bold.cyan(` ${t("skillsSync:title.list")}`));
|
|
670
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
671
|
+
console.log("");
|
|
672
|
+
try {
|
|
673
|
+
const response = await listCloudSkills({ privacy: options.privacy });
|
|
674
|
+
if (!response.success) {
|
|
675
|
+
throw new Error(response.error || "Failed to list cloud skills");
|
|
676
|
+
}
|
|
677
|
+
const skills = response.data?.skills || [];
|
|
678
|
+
if (skills.length === 0) {
|
|
679
|
+
console.log(ansis.yellow(` ${t("skillsSync:message.noCloudSkills")}`));
|
|
680
|
+
return;
|
|
681
|
+
}
|
|
682
|
+
console.log(ansis.bold.green(` ${t("skillsSync:message.foundCloudSkills", { count: skills.length })}
|
|
683
|
+
`));
|
|
684
|
+
for (const skill of skills) {
|
|
685
|
+
const privacyBadge = getPrivacyBadge(skill.privacy);
|
|
686
|
+
console.log(`${ansis.bold(` ${skill.name}`) + ansis.dim(` v${skill.version}`)} ${privacyBadge}`);
|
|
687
|
+
console.log(ansis.dim(` ${skill.metadata.description}`));
|
|
688
|
+
if (skill.metadata.tags && skill.metadata.tags.length > 0) {
|
|
689
|
+
const tags = skill.metadata.tags.map((tag) => ansis.bgGray.white(` ${tag} `)).join(" ");
|
|
690
|
+
console.log(` ${tags}`);
|
|
691
|
+
}
|
|
692
|
+
console.log(ansis.dim(` ${t("skillsSync:label.author")}: ${skill.metadata.author}`));
|
|
693
|
+
console.log(ansis.dim(` ${t("skillsSync:label.updated")}: ${new Date(skill.updatedAt).toLocaleString()}`));
|
|
694
|
+
console.log("");
|
|
695
|
+
}
|
|
696
|
+
} catch (error) {
|
|
697
|
+
console.error(ansis.red(`
|
|
698
|
+
${t("skillsSync:error.listFailed")}: ${error}`));
|
|
699
|
+
throw error;
|
|
700
|
+
}
|
|
701
|
+
}
|
|
702
|
+
async function showSyncStatus(options = {}) {
|
|
703
|
+
const t = getTranslation(options.lang);
|
|
704
|
+
try {
|
|
705
|
+
const syncState = loadSyncState();
|
|
706
|
+
const localSkills = await getLocalSkills();
|
|
707
|
+
const cloudResponse = await listCloudSkills({ privacy: options.privacy });
|
|
708
|
+
const cloudSkills = cloudResponse.success ? cloudResponse.data?.skills || [] : [];
|
|
709
|
+
console.log(ansis.bold(` ${t("skillsSync:label.status")}:`));
|
|
710
|
+
console.log(ansis.dim(` ${t("skillsSync:label.localSkills")}: ${localSkills.length}`));
|
|
711
|
+
console.log(ansis.dim(` ${t("skillsSync:label.cloudSkills")}: ${cloudSkills.length}`));
|
|
712
|
+
console.log(ansis.dim(` ${t("skillsSync:label.lastSync")}: ${new Date(syncState.lastGlobalSync).toLocaleString()}`));
|
|
713
|
+
const states = Object.values(syncState.skills);
|
|
714
|
+
const synced = states.filter((s) => s.status === "synced").length;
|
|
715
|
+
const localAhead = states.filter((s) => s.status === "local_ahead").length;
|
|
716
|
+
const remoteAhead = states.filter((s) => s.status === "remote_ahead").length;
|
|
717
|
+
const conflicts = states.filter((s) => s.status === "conflict").length;
|
|
718
|
+
const localOnly = states.filter((s) => s.status === "local_only").length;
|
|
719
|
+
const remoteOnly = states.filter((s) => s.status === "remote_only").length;
|
|
720
|
+
console.log("");
|
|
721
|
+
console.log(ansis.bold(` ${t("skillsSync:label.syncStates")}:`));
|
|
722
|
+
if (synced > 0)
|
|
723
|
+
console.log(ansis.green(` \u2713 ${t("skillsSync:status.synced")}: ${synced}`));
|
|
724
|
+
if (localAhead > 0)
|
|
725
|
+
console.log(ansis.yellow(` \u2191 ${t("skillsSync:status.localAhead")}: ${localAhead}`));
|
|
726
|
+
if (remoteAhead > 0)
|
|
727
|
+
console.log(ansis.yellow(` \u2193 ${t("skillsSync:status.remoteAhead")}: ${remoteAhead}`));
|
|
728
|
+
if (conflicts > 0)
|
|
729
|
+
console.log(ansis.red(` \u26A0 ${t("skillsSync:status.conflict")}: ${conflicts}`));
|
|
730
|
+
if (localOnly > 0)
|
|
731
|
+
console.log(ansis.blue(` \u2295 ${t("skillsSync:status.localOnly")}: ${localOnly}`));
|
|
732
|
+
if (remoteOnly > 0)
|
|
733
|
+
console.log(ansis.blue(` \u2296 ${t("skillsSync:status.remoteOnly")}: ${remoteOnly}`));
|
|
734
|
+
console.log("");
|
|
735
|
+
} catch (error) {
|
|
736
|
+
console.warn(ansis.yellow(` ${t("skillsSync:warning.statusFailed")}: ${error}`));
|
|
737
|
+
}
|
|
738
|
+
}
|
|
739
|
+
async function skillsSyncMenu(options = {}) {
|
|
740
|
+
const t = getTranslation(options.lang);
|
|
741
|
+
while (true) {
|
|
742
|
+
console.log("");
|
|
743
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
744
|
+
console.log(ansis.bold.cyan(` ${t("skillsSync:menu.title")}`));
|
|
745
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
746
|
+
console.log("");
|
|
747
|
+
const { action } = await inquirer.prompt({
|
|
748
|
+
type: "list",
|
|
749
|
+
name: "action",
|
|
750
|
+
message: t("skillsSync:menu.prompt"),
|
|
751
|
+
choices: [
|
|
752
|
+
{ name: `\u{1F504} ${t("skillsSync:menu.sync")}`, value: "sync" },
|
|
753
|
+
{ name: `\u2191 ${t("skillsSync:menu.push")}`, value: "push" },
|
|
754
|
+
{ name: `\u2193 ${t("skillsSync:menu.pull")}`, value: "pull" },
|
|
755
|
+
{ name: `\u{1F4CB} ${t("skillsSync:menu.list")}`, value: "list" },
|
|
756
|
+
{ name: `\u{1F4CA} ${t("skillsSync:menu.status")}`, value: "status" },
|
|
757
|
+
new inquirer.Separator(),
|
|
758
|
+
{ name: `\u{1F519} ${t("skillsSync:menu.back")}`, value: "back" }
|
|
759
|
+
]
|
|
760
|
+
});
|
|
761
|
+
if (action === "back") {
|
|
762
|
+
break;
|
|
763
|
+
}
|
|
764
|
+
try {
|
|
765
|
+
switch (action) {
|
|
766
|
+
case "sync":
|
|
767
|
+
await syncSkills(options);
|
|
768
|
+
break;
|
|
769
|
+
case "push":
|
|
770
|
+
await pushSkillsCommand(options);
|
|
771
|
+
break;
|
|
772
|
+
case "pull":
|
|
773
|
+
await pullSkillsCommand(options);
|
|
774
|
+
break;
|
|
775
|
+
case "list":
|
|
776
|
+
await listCloudSkillsCommand(options);
|
|
777
|
+
break;
|
|
778
|
+
case "status":
|
|
779
|
+
await showSyncStatus(options);
|
|
780
|
+
break;
|
|
781
|
+
}
|
|
782
|
+
} catch (error) {
|
|
783
|
+
console.error(ansis.red(`
|
|
784
|
+
${t("common.error")}: ${error}`));
|
|
785
|
+
}
|
|
786
|
+
await inquirer.prompt({
|
|
787
|
+
type: "input",
|
|
788
|
+
name: "continue",
|
|
789
|
+
message: t("common.pressEnterToContinue")
|
|
790
|
+
});
|
|
791
|
+
}
|
|
792
|
+
}
|
|
793
|
+
function displaySyncResult(result, lang) {
|
|
794
|
+
const t = getTranslation();
|
|
795
|
+
console.log("");
|
|
796
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
797
|
+
console.log(ansis.bold.cyan(` ${t("skillsSync:result.title")}`));
|
|
798
|
+
console.log(ansis.bold.cyan("\u2501".repeat(60)));
|
|
799
|
+
console.log("");
|
|
800
|
+
if (result.success) {
|
|
801
|
+
console.log(ansis.bold.green(` \u2713 ${t("skillsSync:result.success")}`));
|
|
802
|
+
} else {
|
|
803
|
+
console.log(ansis.bold.red(` \u2717 ${t("skillsSync:result.failed")}`));
|
|
804
|
+
if (result.error) {
|
|
805
|
+
console.log(ansis.red(` ${result.error}`));
|
|
806
|
+
}
|
|
807
|
+
}
|
|
808
|
+
console.log("");
|
|
809
|
+
console.log(ansis.bold(` ${t("skillsSync:result.statistics")}:`));
|
|
810
|
+
console.log(ansis.dim(` ${t("skillsSync:result.total")}: ${result.total}`));
|
|
811
|
+
console.log(ansis.green(` ${t("skillsSync:result.succeeded")}: ${result.succeeded}`));
|
|
812
|
+
if (result.failed > 0) {
|
|
813
|
+
console.log(ansis.red(` ${t("skillsSync:result.failed")}: ${result.failed}`));
|
|
814
|
+
}
|
|
815
|
+
if (result.conflicts > 0) {
|
|
816
|
+
console.log(ansis.yellow(` ${t("skillsSync:result.conflicts")}: ${result.conflicts}`));
|
|
817
|
+
}
|
|
818
|
+
console.log(ansis.blue(` ${t("skillsSync:result.uploaded")}: ${result.uploaded}`));
|
|
819
|
+
console.log(ansis.blue(` ${t("skillsSync:result.downloaded")}: ${result.downloaded}`));
|
|
820
|
+
console.log(ansis.dim(` ${t("skillsSync:result.skipped")}: ${result.skipped}`));
|
|
821
|
+
console.log(ansis.dim(` ${t("skillsSync:result.duration")}: ${(result.durationMs / 1e3).toFixed(2)}s`));
|
|
822
|
+
if (result.failed > 0 || result.conflicts > 0) {
|
|
823
|
+
console.log("");
|
|
824
|
+
console.log(ansis.bold(` ${t("skillsSync:result.details")}:`));
|
|
825
|
+
for (const item of result.results) {
|
|
826
|
+
if (!item.success || item.action === "conflict") {
|
|
827
|
+
const icon = item.success ? "\u26A0" : "\u2717";
|
|
828
|
+
const color = item.success ? ansis.yellow : ansis.red;
|
|
829
|
+
console.log(color(` ${icon} ${item.skillName}`));
|
|
830
|
+
if (item.error) {
|
|
831
|
+
console.log(ansis.dim(` ${item.error}`));
|
|
832
|
+
}
|
|
833
|
+
if (item.action === "conflict") {
|
|
834
|
+
console.log(ansis.dim(` ${t("skillsSync:result.conflictHint")}`));
|
|
835
|
+
}
|
|
836
|
+
}
|
|
837
|
+
}
|
|
838
|
+
}
|
|
839
|
+
console.log("");
|
|
840
|
+
}
|
|
841
|
+
function getPrivacyBadge(privacy) {
|
|
842
|
+
switch (privacy) {
|
|
843
|
+
case "private":
|
|
844
|
+
return ansis.bgRed.white(" PRIVATE ");
|
|
845
|
+
case "team":
|
|
846
|
+
return ansis.bgYellow.black(" TEAM ");
|
|
847
|
+
case "public":
|
|
848
|
+
return ansis.bgGreen.white(" PUBLIC ");
|
|
849
|
+
default:
|
|
850
|
+
return "";
|
|
851
|
+
}
|
|
852
|
+
}
|
|
853
|
+
|
|
854
|
+
export { listCloudSkillsCommand, pullSkillsCommand, pushSkillsCommand, showSyncStatus, skillsSyncMenu, syncSkills };
|