ccjk 13.3.5 → 13.3.7
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/dist/chunks/agent-teams.mjs +7 -5
- package/dist/chunks/agent.mjs +2 -2
- package/dist/chunks/agents.mjs +16 -16
- package/dist/chunks/api-cli.mjs +6 -6
- package/dist/chunks/api-providers.mjs +1 -1
- package/dist/chunks/api.mjs +4 -4
- package/dist/chunks/auto-bootstrap.mjs +1 -1
- package/dist/chunks/auto-fix.mjs +49 -4
- package/dist/chunks/auto-fixer.mjs +7 -5
- package/dist/chunks/auto-init.mjs +9 -7208
- package/dist/chunks/auto-memory-bridge.mjs +9 -3
- package/dist/chunks/auto-updater.mjs +9 -9
- package/dist/chunks/auto-upgrade.mjs +5 -3
- package/dist/chunks/banner.mjs +4 -3
- package/dist/chunks/boost.mjs +118 -62
- package/dist/chunks/ccjk-agents.mjs +3 -3
- package/dist/chunks/ccjk-all.mjs +7 -7
- package/dist/chunks/ccjk-config.mjs +2 -2
- package/dist/chunks/ccjk-hooks.mjs +4 -4
- package/dist/chunks/ccjk-mcp.mjs +5 -5
- package/dist/chunks/ccjk-setup.mjs +5 -5
- package/dist/chunks/ccjk-skills.mjs +5 -5
- package/dist/chunks/ccr.mjs +18 -16
- package/dist/chunks/ccu.mjs +2 -2
- package/dist/chunks/check-updates.mjs +8 -8
- package/dist/chunks/claude-code-config-manager.mjs +11 -8
- package/dist/chunks/claude-code-incremental-manager.mjs +7 -7
- package/dist/chunks/claude-config.mjs +1 -1
- package/dist/chunks/claude-wrapper.mjs +1 -1
- package/dist/chunks/cli-hook.mjs +15 -15
- package/dist/chunks/codex-config-switch.mjs +7 -7
- package/dist/chunks/codex-provider-manager.mjs +7 -7
- package/dist/chunks/codex-uninstaller.mjs +2 -2
- package/dist/chunks/codex.mjs +5 -5
- package/dist/chunks/commands.mjs +2 -2
- package/dist/chunks/commands2.mjs +3 -3
- package/dist/chunks/commit.mjs +2 -2
- package/dist/chunks/completion.mjs +2 -2
- package/dist/chunks/config-consolidator.mjs +2 -2
- package/dist/chunks/config-switch.mjs +8 -8
- package/dist/chunks/config.mjs +6 -5
- package/dist/chunks/config2.mjs +5 -5
- package/dist/chunks/config3.mjs +4 -4
- package/dist/chunks/constants.mjs +1 -1
- package/dist/chunks/context-opt.mjs +92 -90
- package/dist/chunks/context.mjs +659 -0
- package/dist/chunks/dashboard.mjs +14 -9
- package/dist/chunks/doctor.mjs +4 -4
- package/dist/chunks/eval.mjs +502 -0
- package/dist/chunks/evolution.mjs +46 -39
- package/dist/chunks/health-alerts.mjs +9 -9
- package/dist/chunks/help.mjs +1 -1
- package/dist/chunks/hook-installer.mjs +6 -3
- package/dist/chunks/index.mjs +23 -0
- package/dist/chunks/index10.mjs +634 -571
- package/dist/chunks/index11.mjs +1061 -569
- package/dist/chunks/index12.mjs +914 -1076
- package/dist/chunks/index13.mjs +136 -951
- package/dist/chunks/index14.mjs +209 -185
- package/dist/chunks/index2.mjs +19 -24
- package/dist/chunks/index3.mjs +19085 -12
- package/dist/chunks/index4.mjs +16 -19092
- package/dist/chunks/index5.mjs +7602 -16
- package/dist/chunks/index6.mjs +159 -7590
- package/dist/chunks/index7.mjs +1602 -171
- package/dist/chunks/index8.mjs +19 -1602
- package/dist/chunks/index9.mjs +612 -15
- package/dist/chunks/init.mjs +26 -19
- package/dist/chunks/installer.mjs +5 -5
- package/dist/chunks/installer2.mjs +2 -2
- package/dist/chunks/intent-engine.mjs +1 -1
- package/dist/chunks/interview.mjs +4 -4
- package/dist/chunks/manager.mjs +1 -1
- package/dist/chunks/marketplace.mjs +2 -2
- package/dist/chunks/mcp-cli.mjs +12 -12
- package/dist/chunks/mcp.mjs +8 -8
- package/dist/chunks/memory.mjs +8 -8
- package/dist/chunks/menu-hierarchical.mjs +24 -22
- package/dist/chunks/menu.mjs +27 -22
- package/dist/chunks/metrics-display.mjs +2 -2
- package/dist/chunks/migrator.mjs +1 -1
- package/dist/chunks/monitor.mjs +2 -2
- package/dist/chunks/notification.mjs +6 -6
- package/dist/chunks/onboarding-wizard.mjs +6 -5
- package/dist/chunks/onboarding.mjs +4 -4
- package/dist/chunks/package.mjs +1 -1
- package/dist/chunks/paradigm.mjs +2 -2
- package/dist/chunks/permission-manager.mjs +2 -2
- package/dist/chunks/permissions.mjs +3 -3
- package/dist/chunks/persistence-manager.mjs +19 -12
- package/dist/chunks/persistence.mjs +5 -3
- package/dist/chunks/plugin.mjs +2 -2
- package/dist/chunks/prompts.mjs +5 -5
- package/dist/chunks/providers.mjs +2 -2
- package/dist/chunks/quick-actions.mjs +7 -6
- package/dist/chunks/quick-provider.mjs +5 -4
- package/dist/chunks/quick-setup.mjs +20 -15
- package/dist/chunks/remote.mjs +15 -16
- package/dist/chunks/{convoy-manager.mjs → session-manager.mjs} +1129 -1095
- package/dist/chunks/session.mjs +2 -2
- package/dist/chunks/sessions.mjs +3 -3
- package/dist/chunks/silent-updater.mjs +1 -1
- package/dist/chunks/simple-config.mjs +2 -2
- package/dist/chunks/skill2.mjs +3 -3
- package/dist/chunks/skills-sync.mjs +5 -5
- package/dist/chunks/skills.mjs +3 -3
- package/dist/chunks/slash-commands.mjs +9 -8
- package/dist/chunks/smart-defaults.mjs +9 -5
- package/dist/chunks/startup.mjs +1 -1
- package/dist/chunks/stats.mjs +2 -2
- package/dist/chunks/status.mjs +37 -22
- package/dist/chunks/team.mjs +3 -3
- package/dist/chunks/thinking.mjs +4 -4
- package/dist/chunks/trace.mjs +2 -2
- package/dist/chunks/uninstall.mjs +9 -9
- package/dist/chunks/update.mjs +14 -11
- package/dist/chunks/upgrade-manager.mjs +3 -3
- package/dist/chunks/upgrade.mjs +25 -9
- package/dist/chunks/version-checker.mjs +4 -4
- package/dist/chunks/vim.mjs +3 -3
- package/dist/chunks/workflows.mjs +1 -1
- package/dist/chunks/wsl.mjs +1 -1
- package/dist/chunks/zero-config.mjs +4 -4
- package/dist/cli.mjs +60 -26
- package/dist/index.d.mts +4392 -4392
- package/dist/index.d.ts +4392 -4392
- package/dist/index.mjs +4314 -4314
- package/dist/shared/{ccjk.DcKLglJQ.mjs → ccjk.BIxuVL3_.mjs} +2 -2
- package/dist/shared/{ccjk.DJdmgr2d.mjs → ccjk.BJMRY2Ra.mjs} +5 -3
- package/dist/shared/{ccjk.B1TwPltj.mjs → ccjk.BOu1yav7.mjs} +3 -2
- package/dist/shared/{ccjk.mJpVRDZ8.mjs → ccjk.BWFpnOr3.mjs} +1 -1
- package/dist/shared/{ccjk.BfIpomdz.mjs → ccjk.CHUEFqmw.mjs} +3 -2
- package/dist/shared/{ccjk.CqdbaXqU.mjs → ccjk.CLUL0pAV.mjs} +9 -5
- package/dist/shared/{ccjk.Cot9p9_n.mjs → ccjk.Cjj8SVrn.mjs} +1 -1
- package/dist/shared/{ccjk.CfrpIIKy.mjs → ccjk.Crd_nEfj.mjs} +38 -20
- package/dist/shared/{ccjk.DCw2WnZU.mjs → ccjk.CvChMYvB.mjs} +1 -1
- package/dist/shared/{ccjk.CXzjn01x.mjs → ccjk.D8ZLYSZZ.mjs} +1 -1
- package/dist/shared/{ccjk.BrPUmTqm.mjs → ccjk.DJuyfrlL.mjs} +164 -82
- package/dist/shared/{ccjk.DHXfsrwn.mjs → ccjk.DRfdq6yl.mjs} +4 -4
- package/dist/shared/{ccjk.DXRAZcix.mjs → ccjk.DScm_NnL.mjs} +8 -4
- package/dist/shared/{ccjk.XsJWJuQP.mjs → ccjk.DfZKjHvG.mjs} +6 -128
- package/dist/shared/{ccjk.BFxsJM0k.mjs → ccjk.DwSebGy0.mjs} +4 -3
- package/dist/shared/ccjk.DxWqH-EF.mjs +170 -0
- package/dist/shared/{ccjk.Cwa_FiTX.mjs → ccjk.I6IuYdc_.mjs} +2 -2
- package/dist/shared/{ccjk.DpstNaeR.mjs → ccjk.KpFl2RDA.mjs} +3 -3
- package/dist/shared/{ccjk.dYDLfmph.mjs → ccjk._dESH4Rk.mjs} +1 -1
- package/dist/shared/{ccjk.BxSmJ8B7.mjs → ccjk.wLJHO0Af.mjs} +2 -1
- package/package.json +65 -67
- package/dist/chunks/index15.mjs +0 -218
- package/dist/shared/{ccjk.c-ETfBZ_.mjs → ccjk.eIn-g1yI.mjs} +96 -96
package/dist/chunks/index12.mjs
CHANGED
|
@@ -1,1171 +1,1009 @@
|
|
|
1
|
-
import
|
|
2
|
-
import
|
|
3
|
-
import
|
|
4
|
-
import
|
|
1
|
+
import a from './index2.mjs';
|
|
2
|
+
import ora from './index7.mjs';
|
|
3
|
+
import fs from 'node:fs/promises';
|
|
4
|
+
import os__default from 'node:os';
|
|
5
|
+
import path__default from 'node:path';
|
|
6
|
+
import { exec } from 'node:child_process';
|
|
7
|
+
import { promisify } from 'node:util';
|
|
8
|
+
import '../shared/ccjk.BAGoDD49.mjs';
|
|
9
|
+
import 'node:process';
|
|
10
|
+
import '../shared/ccjk.Cjgrln_h.mjs';
|
|
11
|
+
import '../shared/ccjk.DeWpAShp.mjs';
|
|
5
12
|
|
|
6
|
-
function
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
if (since) {
|
|
10
|
-
gitCmd += ` ${since}..${until || "HEAD"}`;
|
|
11
|
-
}
|
|
12
|
-
gitCmd += ` -n ${limit}`;
|
|
13
|
-
try {
|
|
14
|
-
const output = execSync(gitCmd, { cwd, encoding: "utf-8" });
|
|
15
|
-
return parseGitLog(output);
|
|
16
|
-
} catch {
|
|
17
|
-
return [];
|
|
13
|
+
function getCcjkConfigDir() {
|
|
14
|
+
if (process.env.CCJK_CONFIG_DIR) {
|
|
15
|
+
return process.env.CCJK_CONFIG_DIR;
|
|
18
16
|
}
|
|
17
|
+
return path__default.join(os__default.homedir(), ".ccjk");
|
|
18
|
+
}
|
|
19
|
+
function getInstallPath(pluginType, name) {
|
|
20
|
+
const configDir = getCcjkConfigDir();
|
|
21
|
+
const typeDir = {
|
|
22
|
+
skill: "skills",
|
|
23
|
+
mcp: "mcp-servers",
|
|
24
|
+
agent: "agents",
|
|
25
|
+
hook: "hooks"
|
|
26
|
+
}[pluginType];
|
|
27
|
+
return path__default.join(configDir, typeDir, name);
|
|
19
28
|
}
|
|
20
|
-
function
|
|
21
|
-
|
|
22
|
-
const entries =
|
|
29
|
+
async function copyDirectory(src, dest) {
|
|
30
|
+
await fs.mkdir(dest, { recursive: true });
|
|
31
|
+
const entries = await fs.readdir(src, { withFileTypes: true });
|
|
23
32
|
for (const entry of entries) {
|
|
24
|
-
const
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
continue;
|
|
31
|
-
const [hash, shortHash, message, author, date] = parts;
|
|
32
|
-
const files = fileLines.filter((f) => f.trim());
|
|
33
|
-
if (isFixCommit(message)) {
|
|
34
|
-
commits.push({
|
|
35
|
-
hash,
|
|
36
|
-
shortHash,
|
|
37
|
-
message,
|
|
38
|
-
author,
|
|
39
|
-
date,
|
|
40
|
-
files
|
|
41
|
-
});
|
|
33
|
+
const srcPath = path__default.join(src, entry.name);
|
|
34
|
+
const destPath = path__default.join(dest, entry.name);
|
|
35
|
+
if (entry.isDirectory()) {
|
|
36
|
+
await copyDirectory(srcPath, destPath);
|
|
37
|
+
} else {
|
|
38
|
+
await fs.copyFile(srcPath, destPath);
|
|
42
39
|
}
|
|
43
40
|
}
|
|
44
|
-
return commits;
|
|
45
|
-
}
|
|
46
|
-
function isFixCommit(message) {
|
|
47
|
-
const fixPatterns = [
|
|
48
|
-
/^fix[(:]/i,
|
|
49
|
-
/^bugfix[(:]/i,
|
|
50
|
-
/^hotfix[(:]/i,
|
|
51
|
-
/\bfix\b/i,
|
|
52
|
-
/\bbug\b/i,
|
|
53
|
-
/\brepair\b/i,
|
|
54
|
-
/\bresolve\b/i,
|
|
55
|
-
/\bcorrect\b/i,
|
|
56
|
-
/修复/,
|
|
57
|
-
/修正/,
|
|
58
|
-
/解决/,
|
|
59
|
-
/bug/i
|
|
60
|
-
];
|
|
61
|
-
return fixPatterns.some((p) => p.test(message));
|
|
62
41
|
}
|
|
63
|
-
function
|
|
64
|
-
const
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
solution,
|
|
76
|
-
preventionSuggestions,
|
|
77
|
-
relatedPostmortems: []
|
|
78
|
-
};
|
|
42
|
+
async function downloadFile(url, destPath) {
|
|
43
|
+
const response = await fetch(url, {
|
|
44
|
+
headers: {
|
|
45
|
+
"User-Agent": "ccjk-cli"
|
|
46
|
+
},
|
|
47
|
+
redirect: "follow"
|
|
48
|
+
});
|
|
49
|
+
if (!response.ok) {
|
|
50
|
+
throw new Error(`Failed to download: ${response.statusText}`);
|
|
51
|
+
}
|
|
52
|
+
const buffer = Buffer.from(await response.arrayBuffer());
|
|
53
|
+
await fs.writeFile(destPath, buffer);
|
|
79
54
|
}
|
|
80
|
-
|
|
55
|
+
|
|
56
|
+
async function installFromGitHub(sourceInfo, pluginType, options = {}) {
|
|
57
|
+
const { force = false, dryRun = false } = options;
|
|
58
|
+
const { owner, repo, ref = "main", subpath } = sourceInfo;
|
|
81
59
|
try {
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
60
|
+
const repoInfo = await fetchRepoInfo(owner, repo);
|
|
61
|
+
const defaultBranch = repoInfo.default_branch || "main";
|
|
62
|
+
const actualRef = ref || defaultBranch;
|
|
63
|
+
const installPath = getInstallPath(pluginType, repo);
|
|
64
|
+
if (!force) {
|
|
65
|
+
try {
|
|
66
|
+
await fs.access(installPath);
|
|
67
|
+
return {
|
|
68
|
+
success: false,
|
|
69
|
+
source: sourceInfo.originalUrl,
|
|
70
|
+
sourceType: "github",
|
|
71
|
+
pluginType,
|
|
72
|
+
error: `Plugin already exists at ${installPath}. Use --force to overwrite.`
|
|
73
|
+
};
|
|
74
|
+
} catch {
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
if (dryRun) {
|
|
78
|
+
const files = await listRepoFiles(owner, repo, actualRef, subpath);
|
|
79
|
+
return {
|
|
80
|
+
success: true,
|
|
81
|
+
source: sourceInfo.originalUrl,
|
|
82
|
+
sourceType: "github",
|
|
83
|
+
pluginType,
|
|
84
|
+
installedPath: installPath,
|
|
85
|
+
details: {
|
|
86
|
+
name: repo,
|
|
87
|
+
version: actualRef,
|
|
88
|
+
description: repoInfo.description || void 0,
|
|
89
|
+
files: files.slice(0, 20)
|
|
90
|
+
}
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
const tempDir = await downloadRepo(owner, repo, actualRef);
|
|
94
|
+
const sourcePath = subpath ? path__default.join(tempDir, subpath) : tempDir;
|
|
95
|
+
await fs.mkdir(path__default.dirname(installPath), { recursive: true });
|
|
96
|
+
await copyDirectory(sourcePath, installPath);
|
|
97
|
+
await fs.rm(tempDir, { recursive: true, force: true });
|
|
98
|
+
const installedFiles = await listInstalledFiles$1(installPath);
|
|
99
|
+
return {
|
|
100
|
+
success: true,
|
|
101
|
+
source: sourceInfo.originalUrl,
|
|
102
|
+
sourceType: "github",
|
|
103
|
+
pluginType,
|
|
104
|
+
installedPath: installPath,
|
|
105
|
+
details: {
|
|
106
|
+
name: repo,
|
|
107
|
+
version: actualRef,
|
|
108
|
+
description: repoInfo.description || void 0,
|
|
109
|
+
files: installedFiles
|
|
110
|
+
}
|
|
111
|
+
};
|
|
112
|
+
} catch (error) {
|
|
113
|
+
return {
|
|
114
|
+
success: false,
|
|
115
|
+
source: sourceInfo.originalUrl,
|
|
116
|
+
sourceType: "github",
|
|
117
|
+
pluginType,
|
|
118
|
+
error: error instanceof Error ? error.message : String(error)
|
|
119
|
+
};
|
|
90
120
|
}
|
|
91
121
|
}
|
|
92
|
-
function
|
|
93
|
-
const
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
patterns: [
|
|
99
|
-
/null|undefined|cannot read|typeerror/,
|
|
100
|
-
/类型|空值|未定义/,
|
|
101
|
-
/optional chaining|\?\./,
|
|
102
|
-
/strict.*null|null.*check/
|
|
103
|
-
]
|
|
104
|
-
},
|
|
105
|
-
{
|
|
106
|
-
category: "error-handling",
|
|
107
|
-
patterns: [
|
|
108
|
-
/try.*catch|exception|throw|error.*handling/,
|
|
109
|
-
/unhandled.*rejection|promise.*reject/,
|
|
110
|
-
/异常|错误处理|捕获/
|
|
111
|
-
]
|
|
112
|
-
},
|
|
113
|
-
{
|
|
114
|
-
category: "performance",
|
|
115
|
-
patterns: [
|
|
116
|
-
/performance|slow|timeout|memory|leak/,
|
|
117
|
-
/optimize|optimization|cache/,
|
|
118
|
-
/性能|优化|缓存|内存/
|
|
119
|
-
]
|
|
120
|
-
},
|
|
121
|
-
{
|
|
122
|
-
category: "security",
|
|
123
|
-
patterns: [
|
|
124
|
-
/security|xss|csrf|injection|auth/,
|
|
125
|
-
/vulnerability|exploit|sanitize/,
|
|
126
|
-
/安全|漏洞|注入|认证/
|
|
127
|
-
]
|
|
128
|
-
},
|
|
129
|
-
{
|
|
130
|
-
category: "race-condition",
|
|
131
|
-
patterns: [
|
|
132
|
-
/race.*condition|concurrent|async.*await/,
|
|
133
|
-
/deadlock|mutex|lock/,
|
|
134
|
-
/竞态|并发|死锁/
|
|
135
|
-
]
|
|
136
|
-
},
|
|
137
|
-
{
|
|
138
|
-
category: "logic-error",
|
|
139
|
-
patterns: [
|
|
140
|
-
/logic|incorrect|wrong.*result/,
|
|
141
|
-
/逻辑|错误结果|计算错误/
|
|
142
|
-
]
|
|
143
|
-
},
|
|
144
|
-
{
|
|
145
|
-
category: "api-misuse",
|
|
146
|
-
patterns: [
|
|
147
|
-
/api.*usage|incorrect.*call|wrong.*parameter/,
|
|
148
|
-
/接口|调用错误|参数错误/
|
|
149
|
-
]
|
|
150
|
-
},
|
|
151
|
-
{
|
|
152
|
-
category: "configuration",
|
|
153
|
-
patterns: [
|
|
154
|
-
/config|setting|environment|env/,
|
|
155
|
-
/配置|环境|设置/
|
|
156
|
-
]
|
|
157
|
-
},
|
|
158
|
-
{
|
|
159
|
-
category: "dependency",
|
|
160
|
-
patterns: [
|
|
161
|
-
/dependency|package|version|upgrade/,
|
|
162
|
-
/依赖|版本|升级/
|
|
163
|
-
]
|
|
122
|
+
async function fetchRepoInfo(owner, repo) {
|
|
123
|
+
const url = `https://api.github.com/repos/${owner}/${repo}`;
|
|
124
|
+
const response = await fetch(url, {
|
|
125
|
+
headers: {
|
|
126
|
+
"Accept": "application/vnd.github.v3+json",
|
|
127
|
+
"User-Agent": "ccjk-cli"
|
|
164
128
|
}
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
if (
|
|
168
|
-
|
|
129
|
+
});
|
|
130
|
+
if (!response.ok) {
|
|
131
|
+
if (response.status === 404) {
|
|
132
|
+
throw new Error(`Repository not found: ${owner}/${repo}`);
|
|
169
133
|
}
|
|
134
|
+
throw new Error(`Failed to fetch repository info: ${response.statusText}`);
|
|
170
135
|
}
|
|
171
|
-
return
|
|
136
|
+
return response.json();
|
|
172
137
|
}
|
|
173
|
-
function
|
|
174
|
-
const
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
}
|
|
182
|
-
if (
|
|
183
|
-
return
|
|
138
|
+
async function listRepoFiles(owner, repo, ref, subpath) {
|
|
139
|
+
const treePath = subpath || "";
|
|
140
|
+
const url = `https://api.github.com/repos/${owner}/${repo}/git/trees/${ref}?recursive=1`;
|
|
141
|
+
const response = await fetch(url, {
|
|
142
|
+
headers: {
|
|
143
|
+
"Accept": "application/vnd.github.v3+json",
|
|
144
|
+
"User-Agent": "ccjk-cli"
|
|
145
|
+
}
|
|
146
|
+
});
|
|
147
|
+
if (!response.ok) {
|
|
148
|
+
return [];
|
|
184
149
|
}
|
|
185
|
-
|
|
150
|
+
const data = await response.json();
|
|
151
|
+
const files = (data.tree || []).filter((item) => item.type === "blob").map((item) => item.path).filter((filePath) => !treePath || filePath.startsWith(treePath));
|
|
152
|
+
return files;
|
|
186
153
|
}
|
|
187
|
-
function
|
|
188
|
-
const
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
154
|
+
async function downloadRepo(owner, repo, ref) {
|
|
155
|
+
const zipUrl = `https://github.com/${owner}/${repo}/archive/${ref}.zip`;
|
|
156
|
+
const tempDir = path__default.join(os__default.tmpdir(), `ccjk-${repo}-${Date.now()}`);
|
|
157
|
+
const zipPath = path__default.join(tempDir, "repo.zip");
|
|
158
|
+
await fs.mkdir(tempDir, { recursive: true });
|
|
159
|
+
await downloadFile(zipUrl, zipPath);
|
|
160
|
+
const extractedDir = await extractZip(zipPath, tempDir);
|
|
161
|
+
await fs.unlink(zipPath);
|
|
162
|
+
return extractedDir;
|
|
163
|
+
}
|
|
164
|
+
async function extractZip(zipPath, destDir) {
|
|
165
|
+
const { exec } = await import('node:child_process');
|
|
166
|
+
const { promisify } = await import('node:util');
|
|
167
|
+
const execAsync = promisify(exec);
|
|
168
|
+
try {
|
|
169
|
+
await execAsync(`unzip -q "${zipPath}" -d "${destDir}"`);
|
|
170
|
+
} catch {
|
|
171
|
+
try {
|
|
172
|
+
await execAsync(`tar -xf "${zipPath}" -C "${destDir}"`);
|
|
173
|
+
} catch {
|
|
174
|
+
throw new Error("Failed to extract archive. Please ensure unzip or tar is installed.");
|
|
200
175
|
}
|
|
201
176
|
}
|
|
202
|
-
const
|
|
203
|
-
const
|
|
204
|
-
if (
|
|
205
|
-
|
|
177
|
+
const entries = await fs.readdir(destDir);
|
|
178
|
+
const extractedDir = entries.find((entry) => !entry.endsWith(".zip"));
|
|
179
|
+
if (!extractedDir) {
|
|
180
|
+
throw new Error("Failed to find extracted directory");
|
|
206
181
|
}
|
|
207
|
-
return
|
|
182
|
+
return path__default.join(destDir, extractedDir);
|
|
208
183
|
}
|
|
209
|
-
function
|
|
210
|
-
const
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
184
|
+
async function listInstalledFiles$1(dir) {
|
|
185
|
+
const files = [];
|
|
186
|
+
async function walk(currentDir, prefix = "") {
|
|
187
|
+
const entries = await fs.readdir(currentDir, { withFileTypes: true });
|
|
188
|
+
for (const entry of entries) {
|
|
189
|
+
const relativePath = prefix ? `${prefix}/${entry.name}` : entry.name;
|
|
190
|
+
if (entry.isDirectory()) {
|
|
191
|
+
await walk(path__default.join(currentDir, entry.name), relativePath);
|
|
192
|
+
} else {
|
|
193
|
+
files.push(relativePath);
|
|
194
|
+
}
|
|
195
|
+
}
|
|
216
196
|
}
|
|
217
|
-
|
|
197
|
+
await walk(dir);
|
|
198
|
+
return files;
|
|
218
199
|
}
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
"\u63D0\u4F9B\u9ED8\u8BA4\u914D\u7F6E",
|
|
266
|
-
"\u6587\u6863\u5316\u6240\u6709\u914D\u7F6E\u9879",
|
|
267
|
-
"\u5B9E\u73B0\u914D\u7F6E\u70ED\u91CD\u8F7D"
|
|
268
|
-
],
|
|
269
|
-
"dependency": [
|
|
270
|
-
"\u9501\u5B9A\u4F9D\u8D56\u7248\u672C",
|
|
271
|
-
"\u5B9A\u671F\u66F4\u65B0\u4F9D\u8D56",
|
|
272
|
-
"\u4F7F\u7528\u4F9D\u8D56\u626B\u63CF\u5DE5\u5177",
|
|
273
|
-
"\u6D4B\u8BD5\u4F9D\u8D56\u5347\u7EA7"
|
|
274
|
-
],
|
|
275
|
-
"memory-leak": [
|
|
276
|
-
"\u5B9E\u73B0\u8D44\u6E90\u6E05\u7406",
|
|
277
|
-
"\u4F7F\u7528\u5185\u5B58\u5206\u6790\u5DE5\u5177",
|
|
278
|
-
"\u907F\u514D\u5FAA\u73AF\u5F15\u7528",
|
|
279
|
-
"\u5B9A\u671F\u8FDB\u884C\u5185\u5B58\u6D4B\u8BD5"
|
|
280
|
-
],
|
|
281
|
-
"other": [
|
|
282
|
-
"\u589E\u52A0\u6D4B\u8BD5\u8986\u76D6",
|
|
283
|
-
"\u5B9E\u65BD\u4EE3\u7801\u5BA1\u67E5",
|
|
284
|
-
"\u5B8C\u5584\u6587\u6863"
|
|
285
|
-
]
|
|
286
|
-
};
|
|
287
|
-
return suggestions[bugType] || suggestions.other;
|
|
288
|
-
}
|
|
289
|
-
function generatePostmortem(analyses, existingIds) {
|
|
290
|
-
const grouped = groupByCategory(analyses);
|
|
291
|
-
const reports = [];
|
|
292
|
-
let nextId = getNextId(existingIds);
|
|
293
|
-
for (const [category, categoryAnalyses] of Object.entries(grouped)) {
|
|
294
|
-
const merged = mergeAnalyses(categoryAnalyses);
|
|
295
|
-
for (const analysis of merged) {
|
|
296
|
-
const id = `PM-${String(nextId++).padStart(3, "0")}`;
|
|
297
|
-
const report = {
|
|
298
|
-
id,
|
|
299
|
-
title: generateTitle(analysis),
|
|
300
|
-
severity: analysis.severity,
|
|
301
|
-
category: analysis.bugType,
|
|
302
|
-
status: "active",
|
|
303
|
-
createdAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
304
|
-
updatedAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
305
|
-
relatedCommits: [analysis.commit],
|
|
306
|
-
affectedVersions: {
|
|
307
|
-
from: "unknown",
|
|
308
|
-
to: "unknown"
|
|
309
|
-
},
|
|
310
|
-
description: generateDescription(analysis),
|
|
311
|
-
rootCause: [analysis.rootCause],
|
|
312
|
-
solution: {
|
|
313
|
-
description: analysis.solution
|
|
314
|
-
},
|
|
315
|
-
preventionMeasures: analysis.preventionSuggestions,
|
|
316
|
-
aiDirectives: generateAiDirectives(analysis),
|
|
317
|
-
detectionPatterns: generateDetectionPatterns(analysis),
|
|
318
|
-
relatedFiles: analysis.commit.files,
|
|
319
|
-
tags: [category, analysis.severity],
|
|
320
|
-
metadata: {
|
|
321
|
-
generatedBy: "ccjk-postmortem",
|
|
322
|
-
version: "1.0.0"
|
|
200
|
+
|
|
201
|
+
async function installFromLocal(sourceInfo, pluginType, options = {}) {
|
|
202
|
+
const { force = false, dryRun = false } = options;
|
|
203
|
+
const { absolutePath, originalPath } = sourceInfo;
|
|
204
|
+
try {
|
|
205
|
+
const stat = await fs.stat(absolutePath);
|
|
206
|
+
const isFile = stat.isFile();
|
|
207
|
+
const isDirectory = stat.isDirectory();
|
|
208
|
+
if (!isFile && !isDirectory) {
|
|
209
|
+
return {
|
|
210
|
+
success: false,
|
|
211
|
+
source: originalPath,
|
|
212
|
+
sourceType: "local",
|
|
213
|
+
pluginType,
|
|
214
|
+
error: `Source path is neither a file nor a directory: ${absolutePath}`
|
|
215
|
+
};
|
|
216
|
+
}
|
|
217
|
+
const pluginName = path__default.basename(absolutePath, path__default.extname(absolutePath));
|
|
218
|
+
const installPath = getInstallPath(pluginType, pluginName);
|
|
219
|
+
if (!force) {
|
|
220
|
+
try {
|
|
221
|
+
await fs.access(installPath);
|
|
222
|
+
return {
|
|
223
|
+
success: false,
|
|
224
|
+
source: originalPath,
|
|
225
|
+
sourceType: "local",
|
|
226
|
+
pluginType,
|
|
227
|
+
error: `Plugin already exists at ${installPath}. Use --force to overwrite.`
|
|
228
|
+
};
|
|
229
|
+
} catch {
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
const files = isFile ? [path__default.basename(absolutePath)] : await listFiles(absolutePath);
|
|
233
|
+
const pluginInfo = await readPluginInfo(absolutePath, isFile);
|
|
234
|
+
if (dryRun) {
|
|
235
|
+
return {
|
|
236
|
+
success: true,
|
|
237
|
+
source: originalPath,
|
|
238
|
+
sourceType: "local",
|
|
239
|
+
pluginType,
|
|
240
|
+
installedPath: installPath,
|
|
241
|
+
details: {
|
|
242
|
+
name: pluginInfo.name || pluginName,
|
|
243
|
+
version: pluginInfo.version,
|
|
244
|
+
description: pluginInfo.description,
|
|
245
|
+
files
|
|
323
246
|
}
|
|
324
247
|
};
|
|
325
|
-
|
|
248
|
+
}
|
|
249
|
+
if (isFile) {
|
|
250
|
+
await fs.mkdir(path__default.dirname(installPath), { recursive: true });
|
|
251
|
+
if (absolutePath.endsWith(".md")) {
|
|
252
|
+
await fs.copyFile(absolutePath, installPath);
|
|
253
|
+
} else {
|
|
254
|
+
await fs.mkdir(installPath, { recursive: true });
|
|
255
|
+
await fs.copyFile(
|
|
256
|
+
absolutePath,
|
|
257
|
+
path__default.join(installPath, path__default.basename(absolutePath))
|
|
258
|
+
);
|
|
259
|
+
}
|
|
260
|
+
} else {
|
|
261
|
+
await fs.mkdir(path__default.dirname(installPath), { recursive: true });
|
|
262
|
+
await copyDirectory(absolutePath, installPath);
|
|
263
|
+
}
|
|
264
|
+
const installedFiles = isFile ? [path__default.basename(absolutePath)] : await listFiles(installPath);
|
|
265
|
+
return {
|
|
266
|
+
success: true,
|
|
267
|
+
source: originalPath,
|
|
268
|
+
sourceType: "local",
|
|
269
|
+
pluginType,
|
|
270
|
+
installedPath: installPath,
|
|
271
|
+
details: {
|
|
272
|
+
name: pluginInfo.name || pluginName,
|
|
273
|
+
version: pluginInfo.version,
|
|
274
|
+
description: pluginInfo.description,
|
|
275
|
+
files: installedFiles
|
|
276
|
+
}
|
|
277
|
+
};
|
|
278
|
+
} catch (error) {
|
|
279
|
+
return {
|
|
280
|
+
success: false,
|
|
281
|
+
source: originalPath,
|
|
282
|
+
sourceType: "local",
|
|
283
|
+
pluginType,
|
|
284
|
+
error: error instanceof Error ? error.message : String(error)
|
|
285
|
+
};
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
async function listFiles(dir) {
|
|
289
|
+
const files = [];
|
|
290
|
+
async function walk(currentDir, prefix = "") {
|
|
291
|
+
const entries = await fs.readdir(currentDir, { withFileTypes: true });
|
|
292
|
+
for (const entry of entries) {
|
|
293
|
+
if (entry.name.startsWith(".") || entry.name === "node_modules") {
|
|
294
|
+
continue;
|
|
295
|
+
}
|
|
296
|
+
const relativePath = prefix ? `${prefix}/${entry.name}` : entry.name;
|
|
297
|
+
if (entry.isDirectory()) {
|
|
298
|
+
await walk(path__default.join(currentDir, entry.name), relativePath);
|
|
299
|
+
} else {
|
|
300
|
+
files.push(relativePath);
|
|
301
|
+
}
|
|
326
302
|
}
|
|
327
303
|
}
|
|
328
|
-
|
|
304
|
+
await walk(dir);
|
|
305
|
+
return files;
|
|
329
306
|
}
|
|
330
|
-
function
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
307
|
+
async function readPluginInfo(sourcePath, isFile) {
|
|
308
|
+
if (isFile) {
|
|
309
|
+
const name = path__default.basename(sourcePath, path__default.extname(sourcePath));
|
|
310
|
+
return { name };
|
|
311
|
+
}
|
|
312
|
+
try {
|
|
313
|
+
const packageJsonPath = path__default.join(sourcePath, "package.json");
|
|
314
|
+
const content = await fs.readFile(packageJsonPath, "utf-8");
|
|
315
|
+
const packageJson = JSON.parse(content);
|
|
316
|
+
return {
|
|
317
|
+
name: packageJson.name,
|
|
318
|
+
version: packageJson.version,
|
|
319
|
+
description: packageJson.description
|
|
320
|
+
};
|
|
321
|
+
} catch {
|
|
322
|
+
}
|
|
323
|
+
const metaFiles = ["SKILL.md", "skill.md", "README.md", "readme.md"];
|
|
324
|
+
for (const metaFile of metaFiles) {
|
|
325
|
+
try {
|
|
326
|
+
const metaPath = path__default.join(sourcePath, metaFile);
|
|
327
|
+
const content = await fs.readFile(metaPath, "utf-8");
|
|
328
|
+
const titleMatch = content.match(/^#\s+(.+)$/m);
|
|
329
|
+
if (titleMatch) {
|
|
330
|
+
return { name: titleMatch[1].trim() };
|
|
331
|
+
}
|
|
332
|
+
} catch {
|
|
336
333
|
}
|
|
337
|
-
grouped[category].push(analysis);
|
|
338
334
|
}
|
|
339
|
-
return
|
|
335
|
+
return {};
|
|
340
336
|
}
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
337
|
+
|
|
338
|
+
const execAsync = promisify(exec);
|
|
339
|
+
async function installFromNpm(sourceInfo, pluginType, options = {}) {
|
|
340
|
+
const { force = false, dryRun = false } = options;
|
|
341
|
+
const { packageName, version } = sourceInfo;
|
|
342
|
+
try {
|
|
343
|
+
const packageInfo = await fetchPackageInfo(packageName, version);
|
|
344
|
+
const packageVersion = version || packageInfo.version;
|
|
345
|
+
const shortName = getShortName(packageName);
|
|
346
|
+
const installPath = getInstallPath(pluginType, shortName);
|
|
347
|
+
if (!force) {
|
|
348
|
+
try {
|
|
349
|
+
await fs.access(installPath);
|
|
350
|
+
return {
|
|
351
|
+
success: false,
|
|
352
|
+
source: packageName,
|
|
353
|
+
sourceType: "npm",
|
|
354
|
+
pluginType,
|
|
355
|
+
error: `Plugin already exists at ${installPath}. Use --force to overwrite.`
|
|
356
|
+
};
|
|
357
|
+
} catch {
|
|
358
|
+
}
|
|
359
|
+
}
|
|
360
|
+
if (dryRun) {
|
|
361
|
+
const files = packageInfo.files || ["(package files)"];
|
|
362
|
+
return {
|
|
363
|
+
success: true,
|
|
364
|
+
source: packageName,
|
|
365
|
+
sourceType: "npm",
|
|
366
|
+
pluginType,
|
|
367
|
+
installedPath: installPath,
|
|
368
|
+
details: {
|
|
369
|
+
name: packageName,
|
|
370
|
+
version: packageVersion,
|
|
371
|
+
description: packageInfo.description,
|
|
372
|
+
files
|
|
373
|
+
}
|
|
374
|
+
};
|
|
375
|
+
}
|
|
376
|
+
if (pluginType === "mcp") {
|
|
377
|
+
await installMcpPackage(packageName, packageVersion, installPath);
|
|
350
378
|
} else {
|
|
351
|
-
|
|
379
|
+
await installGenericPackage(packageName, packageVersion, installPath);
|
|
352
380
|
}
|
|
381
|
+
const installedFiles = await listInstalledFiles(installPath);
|
|
382
|
+
return {
|
|
383
|
+
success: true,
|
|
384
|
+
source: packageName,
|
|
385
|
+
sourceType: "npm",
|
|
386
|
+
pluginType,
|
|
387
|
+
installedPath: installPath,
|
|
388
|
+
details: {
|
|
389
|
+
name: packageName,
|
|
390
|
+
version: packageVersion,
|
|
391
|
+
description: packageInfo.description,
|
|
392
|
+
files: installedFiles
|
|
393
|
+
}
|
|
394
|
+
};
|
|
395
|
+
} catch (error) {
|
|
396
|
+
return {
|
|
397
|
+
success: false,
|
|
398
|
+
source: packageName,
|
|
399
|
+
sourceType: "npm",
|
|
400
|
+
pluginType,
|
|
401
|
+
error: error instanceof Error ? error.message : String(error)
|
|
402
|
+
};
|
|
353
403
|
}
|
|
354
|
-
return merged;
|
|
355
|
-
}
|
|
356
|
-
function calculateFileOverlap(files1, files2) {
|
|
357
|
-
const set1 = new Set(files1);
|
|
358
|
-
const set2 = new Set(files2);
|
|
359
|
-
const intersection = Array.from(set1).filter((f) => set2.has(f));
|
|
360
|
-
const union = /* @__PURE__ */ new Set([...files1, ...files2]);
|
|
361
|
-
return intersection.length / union.size;
|
|
362
404
|
}
|
|
363
|
-
function
|
|
364
|
-
const
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
"performance": "\u6027\u80FD\u95EE\u9898",
|
|
372
|
-
"security": "\u5B89\u5168\u6F0F\u6D1E",
|
|
373
|
-
"race-condition": "\u7ADE\u6001\u6761\u4EF6",
|
|
374
|
-
"logic-error": "\u903B\u8F91\u9519\u8BEF",
|
|
375
|
-
"api-misuse": "API \u4F7F\u7528\u4E0D\u5F53",
|
|
376
|
-
"configuration": "\u914D\u7F6E\u95EE\u9898",
|
|
377
|
-
"dependency": "\u4F9D\u8D56\u95EE\u9898",
|
|
378
|
-
"memory-leak": "\u5185\u5B58\u6CC4\u6F0F",
|
|
379
|
-
"other": "\u5176\u4ED6\u95EE\u9898"
|
|
380
|
-
};
|
|
381
|
-
const baseTitle = categoryTitles[analysis.bugType] || "\u672A\u5206\u7C7B\u95EE\u9898";
|
|
382
|
-
const message = analysis.commit.message;
|
|
383
|
-
const specificPart = message.replace(/^(fix|bugfix|hotfix)[(:]\s*/i, "").split("\n")[0];
|
|
384
|
-
if (specificPart && specificPart.length < 50) {
|
|
385
|
-
return `${baseTitle}: ${specificPart}`;
|
|
405
|
+
async function fetchPackageInfo(packageName, version) {
|
|
406
|
+
const url = version ? `https://registry.npmjs.org/${packageName}/${version}` : `https://registry.npmjs.org/${packageName}/latest`;
|
|
407
|
+
const response = await fetch(url);
|
|
408
|
+
if (!response.ok) {
|
|
409
|
+
if (response.status === 404) {
|
|
410
|
+
throw new Error(`Package not found: ${packageName}`);
|
|
411
|
+
}
|
|
412
|
+
throw new Error(`Failed to fetch package info: ${response.statusText}`);
|
|
386
413
|
}
|
|
387
|
-
|
|
414
|
+
const data = await response.json();
|
|
415
|
+
return {
|
|
416
|
+
version: data.version,
|
|
417
|
+
description: data.description,
|
|
418
|
+
files: data.files
|
|
419
|
+
};
|
|
388
420
|
}
|
|
389
|
-
function
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
**\u5F71\u54CD\u6587\u4EF6**:
|
|
396
|
-
${analysis.commit.files.map((f) => `- ${f}`).join("\n")}
|
|
397
|
-
|
|
398
|
-
**\u6839\u672C\u539F\u56E0**: ${analysis.rootCause}
|
|
399
|
-
`.trim();
|
|
421
|
+
function getShortName(packageName) {
|
|
422
|
+
if (packageName.startsWith("@")) {
|
|
423
|
+
const parts = packageName.split("/");
|
|
424
|
+
return parts[1] || packageName;
|
|
425
|
+
}
|
|
426
|
+
return packageName;
|
|
400
427
|
}
|
|
401
|
-
function
|
|
402
|
-
|
|
403
|
-
const
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
"error-handling": [
|
|
410
|
-
"\u6240\u6709\u5F02\u6B65\u64CD\u4F5C\u5FC5\u987B\u6709\u9519\u8BEF\u5904\u7406",
|
|
411
|
-
"\u63D0\u4F9B\u6709\u610F\u4E49\u7684\u9519\u8BEF\u6D88\u606F",
|
|
412
|
-
"\u5B9E\u73B0\u4F18\u96C5\u964D\u7EA7"
|
|
413
|
-
],
|
|
414
|
-
"performance": [
|
|
415
|
-
"\u907F\u514D\u5728\u5FAA\u73AF\u4E2D\u8FDB\u884C I/O \u64CD\u4F5C",
|
|
416
|
-
"\u4F7F\u7528\u9002\u5F53\u7684\u7F13\u5B58\u7B56\u7565",
|
|
417
|
-
"\u6CE8\u610F\u5927\u6570\u636E\u96C6\u7684\u5904\u7406"
|
|
418
|
-
],
|
|
419
|
-
"security": [
|
|
420
|
-
"\u9A8C\u8BC1\u6240\u6709\u7528\u6237\u8F93\u5165",
|
|
421
|
-
"\u4F7F\u7528\u53C2\u6570\u5316\u67E5\u8BE2",
|
|
422
|
-
"\u4E0D\u8981\u5728\u65E5\u5FD7\u4E2D\u8BB0\u5F55\u654F\u611F\u4FE1\u606F"
|
|
423
|
-
],
|
|
424
|
-
"race-condition": [
|
|
425
|
-
"\u6CE8\u610F\u5F02\u6B65\u64CD\u4F5C\u7684\u6267\u884C\u987A\u5E8F",
|
|
426
|
-
"\u4F7F\u7528\u9002\u5F53\u7684\u540C\u6B65\u673A\u5236",
|
|
427
|
-
"\u907F\u514D\u5171\u4EAB\u53EF\u53D8\u72B6\u6001"
|
|
428
|
-
],
|
|
429
|
-
"logic-error": [
|
|
430
|
-
"\u6DFB\u52A0\u8FB9\u754C\u6761\u4EF6\u68C0\u67E5",
|
|
431
|
-
"\u4F7F\u7528\u65AD\u8A00\u9A8C\u8BC1\u5047\u8BBE",
|
|
432
|
-
"\u7F16\u5199\u5355\u5143\u6D4B\u8BD5\u8986\u76D6\u8FB9\u754C\u60C5\u51B5"
|
|
433
|
-
],
|
|
434
|
-
"api-misuse": [
|
|
435
|
-
"\u67E5\u9605 API \u6587\u6863\u786E\u8BA4\u6B63\u786E\u7528\u6CD5",
|
|
436
|
-
"\u68C0\u67E5\u53C2\u6570\u7C7B\u578B\u548C\u8303\u56F4",
|
|
437
|
-
"\u5904\u7406\u6240\u6709\u53EF\u80FD\u7684\u8FD4\u56DE\u503C"
|
|
438
|
-
],
|
|
439
|
-
"configuration": [
|
|
440
|
-
"\u63D0\u4F9B\u5408\u7406\u7684\u9ED8\u8BA4\u503C",
|
|
441
|
-
"\u9A8C\u8BC1\u914D\u7F6E\u503C\u7684\u6709\u6548\u6027",
|
|
442
|
-
"\u6587\u6863\u5316\u914D\u7F6E\u9009\u9879"
|
|
443
|
-
],
|
|
444
|
-
"dependency": [
|
|
445
|
-
"\u68C0\u67E5\u4F9D\u8D56\u7684\u517C\u5BB9\u6027",
|
|
446
|
-
"\u9605\u8BFB\u66F4\u65B0\u65E5\u5FD7",
|
|
447
|
-
"\u5728\u5347\u7EA7\u524D\u8FDB\u884C\u6D4B\u8BD5"
|
|
448
|
-
],
|
|
449
|
-
"memory-leak": [
|
|
450
|
-
"\u53CA\u65F6\u6E05\u7406\u4E0D\u518D\u4F7F\u7528\u7684\u8D44\u6E90",
|
|
451
|
-
"\u907F\u514D\u5FAA\u73AF\u5F15\u7528",
|
|
452
|
-
"\u4F7F\u7528 WeakMap/WeakSet"
|
|
453
|
-
],
|
|
454
|
-
"other": [
|
|
455
|
-
"\u4ED4\u7EC6\u5BA1\u67E5\u4EE3\u7801\u53D8\u66F4",
|
|
456
|
-
"\u6DFB\u52A0\u9002\u5F53\u7684\u6D4B\u8BD5"
|
|
457
|
-
]
|
|
458
|
-
};
|
|
459
|
-
directives.push(...categoryDirectives[analysis.bugType] || categoryDirectives.other);
|
|
460
|
-
for (const file of analysis.commit.files) {
|
|
461
|
-
if (file.includes("api") || file.includes("service")) {
|
|
462
|
-
directives.push(`\u4FEE\u6539 ${path.basename(file)} \u65F6\u6CE8\u610F API \u517C\u5BB9\u6027`);
|
|
428
|
+
async function installMcpPackage(packageName, version, installPath) {
|
|
429
|
+
await fs.mkdir(installPath, { recursive: true });
|
|
430
|
+
const packageJson = {
|
|
431
|
+
name: `ccjk-mcp-${getShortName(packageName)}`,
|
|
432
|
+
version: "1.0.0",
|
|
433
|
+
private: true,
|
|
434
|
+
dependencies: {
|
|
435
|
+
[packageName]: version
|
|
463
436
|
}
|
|
464
|
-
|
|
465
|
-
|
|
437
|
+
};
|
|
438
|
+
await fs.writeFile(
|
|
439
|
+
path__default.join(installPath, "package.json"),
|
|
440
|
+
JSON.stringify(packageJson, null, 2)
|
|
441
|
+
);
|
|
442
|
+
try {
|
|
443
|
+
await execAsync("npm install", { cwd: installPath });
|
|
444
|
+
} catch (error) {
|
|
445
|
+
try {
|
|
446
|
+
await execAsync("pnpm install", { cwd: installPath });
|
|
447
|
+
} catch {
|
|
448
|
+
throw new Error(
|
|
449
|
+
`Failed to install package. Original error: ${error instanceof Error ? error.message : String(error)}`
|
|
450
|
+
);
|
|
466
451
|
}
|
|
467
452
|
}
|
|
468
|
-
return Array.from(new Set(directives));
|
|
469
453
|
}
|
|
470
|
-
function
|
|
471
|
-
const
|
|
472
|
-
const
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
506
|
-
|
|
507
|
-
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
fileTypes: [".ts", ".tsx", ".js", ".jsx"],
|
|
511
|
-
severity: "medium"
|
|
512
|
-
}
|
|
513
|
-
],
|
|
514
|
-
"security": [
|
|
515
|
-
{
|
|
516
|
-
type: "regex",
|
|
517
|
-
pattern: "innerHTML\\s*=",
|
|
518
|
-
description: "\u76F4\u63A5\u8BBE\u7F6E innerHTML \u53EF\u80FD\u5BFC\u81F4 XSS",
|
|
519
|
-
fileTypes: [".ts", ".tsx", ".js", ".jsx"],
|
|
520
|
-
severity: "high"
|
|
521
|
-
},
|
|
522
|
-
{
|
|
523
|
-
type: "regex",
|
|
524
|
-
pattern: "eval\\s*\\(",
|
|
525
|
-
description: "\u4F7F\u7528 eval \u5B58\u5728\u5B89\u5168\u98CE\u9669",
|
|
526
|
-
fileTypes: [".ts", ".tsx", ".js", ".jsx"],
|
|
527
|
-
severity: "critical"
|
|
528
|
-
}
|
|
529
|
-
],
|
|
530
|
-
"race-condition": [],
|
|
531
|
-
"logic-error": [],
|
|
532
|
-
"api-misuse": [],
|
|
533
|
-
"configuration": [],
|
|
534
|
-
"dependency": [],
|
|
535
|
-
"memory-leak": [
|
|
536
|
-
{
|
|
537
|
-
type: "regex",
|
|
538
|
-
pattern: "addEventListener\\([^)]+\\)(?![\\s\\S]*removeEventListener)",
|
|
539
|
-
description: "\u6DFB\u52A0\u4E8B\u4EF6\u76D1\u542C\u5668\u4F46\u672A\u79FB\u9664",
|
|
540
|
-
fileTypes: [".ts", ".tsx", ".js", ".jsx"],
|
|
541
|
-
severity: "medium"
|
|
454
|
+
async function installGenericPackage(packageName, version, installPath) {
|
|
455
|
+
const registryUrl = `https://registry.npmjs.org/${packageName}/${version}`;
|
|
456
|
+
const response = await fetch(registryUrl);
|
|
457
|
+
if (!response.ok) {
|
|
458
|
+
throw new Error(`Failed to fetch package info: ${response.statusText}`);
|
|
459
|
+
}
|
|
460
|
+
const data = await response.json();
|
|
461
|
+
const tarballUrl = data.dist?.tarball;
|
|
462
|
+
if (!tarballUrl) {
|
|
463
|
+
throw new Error("Failed to get tarball URL");
|
|
464
|
+
}
|
|
465
|
+
const tarballResponse = await fetch(tarballUrl);
|
|
466
|
+
if (!tarballResponse.ok) {
|
|
467
|
+
throw new Error(`Failed to download tarball: ${tarballResponse.statusText}`);
|
|
468
|
+
}
|
|
469
|
+
await fs.mkdir(installPath, { recursive: true });
|
|
470
|
+
const tarballBuffer = Buffer.from(await tarballResponse.arrayBuffer());
|
|
471
|
+
const tarballPath = path__default.join(installPath, "package.tgz");
|
|
472
|
+
await fs.writeFile(tarballPath, tarballBuffer);
|
|
473
|
+
try {
|
|
474
|
+
await execAsync(`tar -xzf package.tgz --strip-components=1`, { cwd: installPath });
|
|
475
|
+
} finally {
|
|
476
|
+
await fs.unlink(tarballPath).catch(() => {
|
|
477
|
+
});
|
|
478
|
+
}
|
|
479
|
+
}
|
|
480
|
+
async function listInstalledFiles(dir) {
|
|
481
|
+
const files = [];
|
|
482
|
+
async function walk(currentDir, prefix = "") {
|
|
483
|
+
try {
|
|
484
|
+
const entries = await fs.readdir(currentDir, { withFileTypes: true });
|
|
485
|
+
for (const entry of entries) {
|
|
486
|
+
if (entry.name === "node_modules")
|
|
487
|
+
continue;
|
|
488
|
+
const relativePath = prefix ? `${prefix}/${entry.name}` : entry.name;
|
|
489
|
+
if (entry.isDirectory()) {
|
|
490
|
+
await walk(path__default.join(currentDir, entry.name), relativePath);
|
|
491
|
+
} else {
|
|
492
|
+
files.push(relativePath);
|
|
493
|
+
}
|
|
542
494
|
}
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
}
|
|
546
|
-
|
|
547
|
-
return
|
|
495
|
+
} catch {
|
|
496
|
+
}
|
|
497
|
+
}
|
|
498
|
+
await walk(dir);
|
|
499
|
+
return files;
|
|
548
500
|
}
|
|
549
|
-
const PostmortemAnalyzer = {
|
|
550
|
-
getFixCommits,
|
|
551
|
-
analyzeFixCommit,
|
|
552
|
-
generatePostmortem
|
|
553
|
-
};
|
|
554
501
|
|
|
555
|
-
|
|
556
|
-
|
|
557
|
-
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
minSyncSeverity: "medium",
|
|
561
|
-
detection: {
|
|
562
|
-
enabled: true,
|
|
563
|
-
excludePatterns: ["node_modules/**", "dist/**", "*.test.*", "*.spec.*"],
|
|
564
|
-
includePatterns: ["src/**/*.ts", "src/**/*.tsx"]
|
|
565
|
-
},
|
|
566
|
-
aiAnalysis: {
|
|
567
|
-
provider: "claude"
|
|
502
|
+
function parseSource(source) {
|
|
503
|
+
const trimmed = source.trim();
|
|
504
|
+
const githubInfo = parseGitHubSource(trimmed);
|
|
505
|
+
if (githubInfo) {
|
|
506
|
+
return githubInfo;
|
|
568
507
|
}
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
|
|
572
|
-
const CLAUDE_MD_SECTION_END = "<!-- POSTMORTEM_END -->";
|
|
573
|
-
class PostmortemManager {
|
|
574
|
-
config;
|
|
575
|
-
projectRoot;
|
|
576
|
-
postmortemDir;
|
|
577
|
-
constructor(projectRoot = process.cwd(), config = {}) {
|
|
578
|
-
this.projectRoot = projectRoot;
|
|
579
|
-
this.config = { ...DEFAULT_CONFIG, ...config };
|
|
580
|
-
this.postmortemDir = path.join(projectRoot, this.config.directory);
|
|
508
|
+
const localInfo = parseLocalSource(trimmed);
|
|
509
|
+
if (localInfo) {
|
|
510
|
+
return localInfo;
|
|
581
511
|
}
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
|
|
589
|
-
|
|
590
|
-
|
|
591
|
-
|
|
592
|
-
|
|
593
|
-
}
|
|
594
|
-
if (commits.length === 0) {
|
|
595
|
-
this.saveIndex(this.createEmptyIndex());
|
|
596
|
-
return { created: 0, directory: this.postmortemDir };
|
|
597
|
-
}
|
|
598
|
-
const analyses = commits.map(
|
|
599
|
-
(commit) => PostmortemAnalyzer.analyzeFixCommit(commit, this.projectRoot)
|
|
600
|
-
);
|
|
601
|
-
const reports = PostmortemAnalyzer.generatePostmortem(analyses, []);
|
|
602
|
-
for (const report of reports) {
|
|
603
|
-
this.saveReport(report);
|
|
604
|
-
}
|
|
605
|
-
this.updateIndex();
|
|
606
|
-
if (this.config.autoSyncToClaudeMd) {
|
|
607
|
-
await this.syncToClaudeMd();
|
|
608
|
-
}
|
|
609
|
-
return { created: reports.length, directory: this.postmortemDir };
|
|
512
|
+
return parseNpmSource(trimmed);
|
|
513
|
+
}
|
|
514
|
+
function parseGitHubSource(source) {
|
|
515
|
+
const prefixMatch = source.match(/^(?:github|gh):([^/]+)\/([^#@/]+)(?:#(.+))?$/);
|
|
516
|
+
if (prefixMatch) {
|
|
517
|
+
return {
|
|
518
|
+
type: "github",
|
|
519
|
+
owner: prefixMatch[1],
|
|
520
|
+
repo: prefixMatch[2],
|
|
521
|
+
ref: prefixMatch[3],
|
|
522
|
+
originalUrl: source
|
|
523
|
+
};
|
|
610
524
|
}
|
|
611
|
-
|
|
612
|
-
|
|
613
|
-
|
|
614
|
-
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
|
|
618
|
-
|
|
619
|
-
|
|
620
|
-
|
|
621
|
-
|
|
622
|
-
|
|
623
|
-
}
|
|
624
|
-
}
|
|
525
|
+
const urlMatch = source.match(
|
|
526
|
+
/^https?:\/\/github\.com\/([^/]+)\/([^/]+?)(?:\.git)?(?:\/tree\/([^/]+))?(?:\/(.+))?$/
|
|
527
|
+
);
|
|
528
|
+
if (urlMatch) {
|
|
529
|
+
return {
|
|
530
|
+
type: "github",
|
|
531
|
+
owner: urlMatch[1],
|
|
532
|
+
repo: urlMatch[2],
|
|
533
|
+
ref: urlMatch[3],
|
|
534
|
+
subpath: urlMatch[4],
|
|
535
|
+
originalUrl: source
|
|
536
|
+
};
|
|
625
537
|
}
|
|
626
|
-
|
|
627
|
-
|
|
628
|
-
|
|
629
|
-
|
|
630
|
-
|
|
631
|
-
|
|
632
|
-
|
|
633
|
-
|
|
634
|
-
|
|
635
|
-
const content = this.renderReportToMarkdown(report);
|
|
636
|
-
fs.writeFileSync(filepath, content, "utf-8");
|
|
637
|
-
const jsonPath = path.join(this.postmortemDir, `${report.id}.json`);
|
|
638
|
-
fs.writeFileSync(jsonPath, JSON.stringify(report, null, 2), "utf-8");
|
|
639
|
-
return filepath;
|
|
538
|
+
const shortMatch = source.match(/^([\w-]+)\/([\w.-]+)(?:#(.+))?$/);
|
|
539
|
+
if (shortMatch && !source.startsWith(".") && !source.startsWith("/") && !source.startsWith("~")) {
|
|
540
|
+
return {
|
|
541
|
+
type: "github",
|
|
542
|
+
owner: shortMatch[1],
|
|
543
|
+
repo: shortMatch[2],
|
|
544
|
+
ref: shortMatch[3],
|
|
545
|
+
originalUrl: source
|
|
546
|
+
};
|
|
640
547
|
}
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
if (
|
|
647
|
-
|
|
648
|
-
}
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
} catch {
|
|
653
|
-
return null;
|
|
548
|
+
return null;
|
|
549
|
+
}
|
|
550
|
+
function parseLocalSource(source) {
|
|
551
|
+
if (source.startsWith("./") || source.startsWith("../") || source.startsWith("/") || source.startsWith("~/")) {
|
|
552
|
+
let absolutePath;
|
|
553
|
+
if (source.startsWith("~/")) {
|
|
554
|
+
absolutePath = path__default.join(os__default.homedir(), source.slice(2));
|
|
555
|
+
} else if (path__default.isAbsolute(source)) {
|
|
556
|
+
absolutePath = source;
|
|
557
|
+
} else {
|
|
558
|
+
absolutePath = path__default.resolve(process.cwd(), source);
|
|
654
559
|
}
|
|
560
|
+
return {
|
|
561
|
+
type: "local",
|
|
562
|
+
absolutePath,
|
|
563
|
+
originalPath: source
|
|
564
|
+
};
|
|
655
565
|
}
|
|
656
|
-
|
|
657
|
-
|
|
658
|
-
|
|
659
|
-
|
|
660
|
-
|
|
661
|
-
return index?.reports || [];
|
|
662
|
-
}
|
|
663
|
-
/**
|
|
664
|
-
* 渲染报告为 Markdown
|
|
665
|
-
*/
|
|
666
|
-
renderReportToMarkdown(report) {
|
|
667
|
-
const severityEmoji = {
|
|
668
|
-
critical: "\u{1F534}",
|
|
669
|
-
high: "\u{1F7E0}",
|
|
670
|
-
medium: "\u{1F7E1}",
|
|
671
|
-
low: "\u{1F7E2}"
|
|
566
|
+
if (/^[a-z]:[\\/]/i.test(source)) {
|
|
567
|
+
return {
|
|
568
|
+
type: "local",
|
|
569
|
+
absolutePath: path__default.resolve(source),
|
|
570
|
+
originalPath: source
|
|
672
571
|
};
|
|
673
|
-
return `# ${report.id}: ${report.title}
|
|
674
|
-
|
|
675
|
-
## \u5143\u6570\u636E
|
|
676
|
-
- **ID**: ${report.id}
|
|
677
|
-
- **\u4E25\u91CD\u7A0B\u5EA6**: ${severityEmoji[report.severity]} ${report.severity.toUpperCase()}
|
|
678
|
-
- **\u7C7B\u522B**: ${report.category}
|
|
679
|
-
- **\u72B6\u6001**: ${report.status}
|
|
680
|
-
- **\u521B\u5EFA\u65F6\u95F4**: ${report.createdAt}
|
|
681
|
-
- **\u66F4\u65B0\u65F6\u95F4**: ${report.updatedAt}
|
|
682
|
-
|
|
683
|
-
## \u76F8\u5173\u63D0\u4EA4
|
|
684
|
-
${report.relatedCommits.map((c) => `- \`${c.shortHash}\` - ${c.message} (${c.author}, ${c.date})`).join("\n")}
|
|
685
|
-
|
|
686
|
-
## \u5F71\u54CD\u7248\u672C
|
|
687
|
-
- **\u4ECE**: ${report.affectedVersions.from}
|
|
688
|
-
- **\u5230**: ${report.affectedVersions.to}
|
|
689
|
-
|
|
690
|
-
## \u95EE\u9898\u63CF\u8FF0
|
|
691
|
-
${report.description}
|
|
692
|
-
|
|
693
|
-
## \u6839\u672C\u539F\u56E0
|
|
694
|
-
${report.rootCause.map((c) => `- ${c}`).join("\n")}
|
|
695
|
-
|
|
696
|
-
## \u4FEE\u590D\u65B9\u6848
|
|
697
|
-
${report.solution.description}
|
|
698
|
-
|
|
699
|
-
${report.solution.codeExample ? `
|
|
700
|
-
### \u4EE3\u7801\u793A\u4F8B
|
|
701
|
-
|
|
702
|
-
**\u274C \u9519\u8BEF\u5199\u6CD5**
|
|
703
|
-
\`\`\`typescript
|
|
704
|
-
${report.solution.codeExample.bad}
|
|
705
|
-
\`\`\`
|
|
706
|
-
|
|
707
|
-
**\u2705 \u6B63\u786E\u5199\u6CD5**
|
|
708
|
-
\`\`\`typescript
|
|
709
|
-
${report.solution.codeExample.good}
|
|
710
|
-
\`\`\`
|
|
711
|
-
` : ""}
|
|
712
|
-
|
|
713
|
-
## \u9884\u9632\u63AA\u65BD
|
|
714
|
-
${report.preventionMeasures.map((m, i) => `${i + 1}. ${m}`).join("\n")}
|
|
715
|
-
|
|
716
|
-
## AI \u5F00\u53D1\u6307\u4EE4
|
|
717
|
-
> \u4EE5\u4E0B\u6307\u4EE4\u4F1A\u81EA\u52A8\u6CE8\u5165\u5230 CLAUDE.md \u4E2D\uFF0C\u6307\u5BFC AI \u5728\u5F00\u53D1\u65F6\u907F\u514D\u7C7B\u4F3C\u95EE\u9898
|
|
718
|
-
|
|
719
|
-
${report.aiDirectives.map((d) => `- ${d}`).join("\n")}
|
|
720
|
-
|
|
721
|
-
## \u68C0\u6D4B\u6A21\u5F0F
|
|
722
|
-
${report.detectionPatterns.length > 0 ? report.detectionPatterns.map((p) => `
|
|
723
|
-
### ${p.description}
|
|
724
|
-
- **\u7C7B\u578B**: ${p.type}
|
|
725
|
-
- **\u6A21\u5F0F**: \`${p.pattern}\`
|
|
726
|
-
- **\u9002\u7528\u6587\u4EF6**: ${p.fileTypes.join(", ")}
|
|
727
|
-
- **\u4E25\u91CD\u7A0B\u5EA6**: ${p.severity}
|
|
728
|
-
`).join("\n") : "\u6682\u65E0\u81EA\u52A8\u68C0\u6D4B\u6A21\u5F0F"}
|
|
729
|
-
|
|
730
|
-
## \u76F8\u5173\u6587\u4EF6
|
|
731
|
-
${report.relatedFiles.map((f) => `- \`${f}\``).join("\n")}
|
|
732
|
-
|
|
733
|
-
## \u6807\u7B7E
|
|
734
|
-
${report.tags.map((t) => `\`${t}\``).join(" ")}
|
|
735
|
-
|
|
736
|
-
---
|
|
737
|
-
*\u7531 CCJK Postmortem System \u81EA\u52A8\u751F\u6210*
|
|
738
|
-
`;
|
|
739
572
|
}
|
|
740
|
-
|
|
741
|
-
|
|
742
|
-
|
|
743
|
-
|
|
744
|
-
|
|
573
|
+
return null;
|
|
574
|
+
}
|
|
575
|
+
function parseNpmSource(source) {
|
|
576
|
+
const npmPrefixMatch = source.match(/^npm:(.+)$/);
|
|
577
|
+
const packageStr = npmPrefixMatch ? npmPrefixMatch[1] : source;
|
|
578
|
+
const scopedMatch = packageStr.match(/^(@[^/]+)\/([^@]+)(?:@(.+))?$/);
|
|
579
|
+
if (scopedMatch) {
|
|
580
|
+
return {
|
|
581
|
+
type: "npm",
|
|
582
|
+
scope: scopedMatch[1],
|
|
583
|
+
packageName: `${scopedMatch[1]}/${scopedMatch[2]}`,
|
|
584
|
+
version: scopedMatch[3]
|
|
585
|
+
};
|
|
745
586
|
}
|
|
746
|
-
|
|
747
|
-
|
|
748
|
-
// ==========================================================================
|
|
749
|
-
/**
|
|
750
|
-
* 创建空索引
|
|
751
|
-
*/
|
|
752
|
-
createEmptyIndex() {
|
|
587
|
+
const versionMatch = packageStr.match(/^([^@]+)@(.+)$/);
|
|
588
|
+
if (versionMatch) {
|
|
753
589
|
return {
|
|
754
|
-
|
|
755
|
-
|
|
756
|
-
|
|
757
|
-
total: 0,
|
|
758
|
-
bySeverity: { critical: 0, high: 0, medium: 0, low: 0 },
|
|
759
|
-
byCategory: {
|
|
760
|
-
"type-safety": 0,
|
|
761
|
-
"error-handling": 0,
|
|
762
|
-
"performance": 0,
|
|
763
|
-
"security": 0,
|
|
764
|
-
"logic-error": 0,
|
|
765
|
-
"race-condition": 0,
|
|
766
|
-
"memory-leak": 0,
|
|
767
|
-
"api-misuse": 0,
|
|
768
|
-
"configuration": 0,
|
|
769
|
-
"dependency": 0,
|
|
770
|
-
"other": 0
|
|
771
|
-
},
|
|
772
|
-
byStatus: { active: 0, resolved: 0, monitoring: 0, archived: 0 }
|
|
773
|
-
},
|
|
774
|
-
reports: []
|
|
590
|
+
type: "npm",
|
|
591
|
+
packageName: versionMatch[1],
|
|
592
|
+
version: versionMatch[2]
|
|
775
593
|
};
|
|
776
594
|
}
|
|
777
|
-
|
|
778
|
-
|
|
779
|
-
|
|
780
|
-
|
|
781
|
-
|
|
782
|
-
|
|
783
|
-
|
|
595
|
+
return {
|
|
596
|
+
type: "npm",
|
|
597
|
+
packageName: packageStr
|
|
598
|
+
};
|
|
599
|
+
}
|
|
600
|
+
function buildGitHubRawUrl(info, filePath) {
|
|
601
|
+
const ref = info.ref || "main";
|
|
602
|
+
return `https://raw.githubusercontent.com/${info.owner}/${info.repo}/${ref}/${filePath}`;
|
|
603
|
+
}
|
|
604
|
+
|
|
605
|
+
async function detectPluginType(sourceInfo) {
|
|
606
|
+
const result = await detectPluginTypeWithConfidence(sourceInfo);
|
|
607
|
+
return result.type;
|
|
608
|
+
}
|
|
609
|
+
async function detectPluginTypeWithConfidence(sourceInfo) {
|
|
610
|
+
switch (sourceInfo.type) {
|
|
611
|
+
case "github":
|
|
612
|
+
return detectFromGitHub(sourceInfo);
|
|
613
|
+
case "npm":
|
|
614
|
+
return detectFromNpm(sourceInfo);
|
|
615
|
+
case "local":
|
|
616
|
+
return detectFromLocal(sourceInfo);
|
|
617
|
+
default:
|
|
618
|
+
return { type: "skill", confidence: "low", reason: "Unknown source type" };
|
|
619
|
+
}
|
|
620
|
+
}
|
|
621
|
+
async function detectFromGitHub(info) {
|
|
622
|
+
const repoNameResult = detectFromRepoName(info.repo);
|
|
623
|
+
if (repoNameResult.confidence === "high") {
|
|
624
|
+
return repoNameResult;
|
|
625
|
+
}
|
|
626
|
+
try {
|
|
627
|
+
const packageJsonUrl = buildGitHubRawUrl(info, "package.json");
|
|
628
|
+
const response = await fetch(packageJsonUrl);
|
|
629
|
+
if (response.ok) {
|
|
630
|
+
const packageJson = await response.json();
|
|
631
|
+
const pkgResult = detectFromPackageJson(packageJson);
|
|
632
|
+
if (pkgResult.confidence !== "low") {
|
|
633
|
+
return pkgResult;
|
|
634
|
+
}
|
|
784
635
|
}
|
|
636
|
+
} catch {
|
|
637
|
+
}
|
|
638
|
+
const filePatterns = [
|
|
639
|
+
{ file: "SKILL.md", type: "skill" },
|
|
640
|
+
{ file: "skill.md", type: "skill" },
|
|
641
|
+
{ file: "AGENT.md", type: "agent" },
|
|
642
|
+
{ file: "agent.md", type: "agent" },
|
|
643
|
+
{ file: "mcp.json", type: "mcp" },
|
|
644
|
+
{ file: "hook.json", type: "hook" }
|
|
645
|
+
];
|
|
646
|
+
for (const pattern of filePatterns) {
|
|
785
647
|
try {
|
|
786
|
-
const
|
|
787
|
-
|
|
648
|
+
const fileUrl = buildGitHubRawUrl(info, pattern.file);
|
|
649
|
+
const response = await fetch(fileUrl, { method: "HEAD" });
|
|
650
|
+
if (response.ok) {
|
|
651
|
+
return {
|
|
652
|
+
type: pattern.type,
|
|
653
|
+
confidence: "high",
|
|
654
|
+
reason: `Found ${pattern.file}`
|
|
655
|
+
};
|
|
656
|
+
}
|
|
788
657
|
} catch {
|
|
789
|
-
return null;
|
|
790
658
|
}
|
|
791
659
|
}
|
|
792
|
-
|
|
793
|
-
|
|
794
|
-
|
|
795
|
-
|
|
796
|
-
|
|
797
|
-
|
|
660
|
+
return repoNameResult.confidence !== "low" ? repoNameResult : { type: "skill", confidence: "low", reason: "Default type" };
|
|
661
|
+
}
|
|
662
|
+
async function detectFromNpm(info) {
|
|
663
|
+
const nameResult = detectFromPackageName(info.packageName);
|
|
664
|
+
if (nameResult.confidence === "high") {
|
|
665
|
+
return nameResult;
|
|
798
666
|
}
|
|
799
|
-
|
|
800
|
-
|
|
801
|
-
|
|
802
|
-
|
|
803
|
-
|
|
804
|
-
|
|
805
|
-
|
|
806
|
-
|
|
807
|
-
const
|
|
808
|
-
|
|
809
|
-
|
|
810
|
-
|
|
811
|
-
index.stats.byCategory[report.category]++;
|
|
812
|
-
index.stats.byStatus[report.status]++;
|
|
813
|
-
index.reports.push({
|
|
814
|
-
id: report.id,
|
|
815
|
-
title: report.title,
|
|
816
|
-
severity: report.severity,
|
|
817
|
-
category: report.category,
|
|
818
|
-
status: report.status,
|
|
819
|
-
createdAt: report.createdAt,
|
|
820
|
-
filePath: file.replace(".json", ".md")
|
|
821
|
-
});
|
|
822
|
-
} catch {
|
|
667
|
+
try {
|
|
668
|
+
const registryUrl = `https://registry.npmjs.org/${info.packageName}`;
|
|
669
|
+
const response = await fetch(registryUrl);
|
|
670
|
+
if (response.ok) {
|
|
671
|
+
const data = await response.json();
|
|
672
|
+
const latestVersion = data["dist-tags"]?.latest;
|
|
673
|
+
if (latestVersion && data.versions?.[latestVersion]) {
|
|
674
|
+
const packageJson = data.versions[latestVersion];
|
|
675
|
+
const pkgResult = detectFromPackageJson(packageJson);
|
|
676
|
+
if (pkgResult.confidence !== "low") {
|
|
677
|
+
return pkgResult;
|
|
678
|
+
}
|
|
823
679
|
}
|
|
824
680
|
}
|
|
825
|
-
|
|
826
|
-
|
|
827
|
-
|
|
828
|
-
|
|
829
|
-
|
|
830
|
-
|
|
831
|
-
|
|
832
|
-
|
|
833
|
-
|
|
834
|
-
return
|
|
681
|
+
} catch {
|
|
682
|
+
}
|
|
683
|
+
return nameResult.confidence !== "low" ? nameResult : { type: "skill", confidence: "low", reason: "Default type" };
|
|
684
|
+
}
|
|
685
|
+
async function detectFromLocal(info) {
|
|
686
|
+
const { absolutePath } = info;
|
|
687
|
+
const dirName = path__default.basename(absolutePath);
|
|
688
|
+
const dirResult = detectFromRepoName(dirName);
|
|
689
|
+
if (dirResult.confidence === "high") {
|
|
690
|
+
return dirResult;
|
|
835
691
|
}
|
|
836
|
-
|
|
837
|
-
|
|
838
|
-
|
|
839
|
-
|
|
840
|
-
|
|
841
|
-
|
|
842
|
-
|
|
843
|
-
const claudeMdPath = path.join(this.projectRoot, "CLAUDE.md");
|
|
844
|
-
const injection = this.generateClaudeMdInjection();
|
|
845
|
-
let content = "";
|
|
846
|
-
if (fs.existsSync(claudeMdPath)) {
|
|
847
|
-
content = fs.readFileSync(claudeMdPath, "utf-8");
|
|
692
|
+
try {
|
|
693
|
+
const packageJsonPath = path__default.join(absolutePath, "package.json");
|
|
694
|
+
const content = await fs.readFile(packageJsonPath, "utf-8");
|
|
695
|
+
const packageJson = JSON.parse(content);
|
|
696
|
+
const pkgResult = detectFromPackageJson(packageJson);
|
|
697
|
+
if (pkgResult.confidence !== "low") {
|
|
698
|
+
return pkgResult;
|
|
848
699
|
}
|
|
849
|
-
|
|
850
|
-
|
|
851
|
-
|
|
852
|
-
|
|
700
|
+
} catch {
|
|
701
|
+
}
|
|
702
|
+
const filePatterns = [
|
|
703
|
+
{ file: "SKILL.md", type: "skill" },
|
|
704
|
+
{ file: "skill.md", type: "skill" },
|
|
705
|
+
{ file: "AGENT.md", type: "agent" },
|
|
706
|
+
{ file: "agent.md", type: "agent" },
|
|
707
|
+
{ file: "mcp.json", type: "mcp" },
|
|
708
|
+
{ file: "hook.json", type: "hook" }
|
|
709
|
+
];
|
|
710
|
+
for (const pattern of filePatterns) {
|
|
711
|
+
try {
|
|
712
|
+
const filePath = path__default.join(absolutePath, pattern.file);
|
|
713
|
+
await fs.access(filePath);
|
|
714
|
+
return {
|
|
715
|
+
type: pattern.type,
|
|
716
|
+
confidence: "high",
|
|
717
|
+
reason: `Found ${pattern.file}`
|
|
718
|
+
};
|
|
719
|
+
} catch {
|
|
853
720
|
}
|
|
854
|
-
const injectionContent = `
|
|
855
|
-
${CLAUDE_MD_SECTION_START}
|
|
856
|
-
${injection.content}
|
|
857
|
-
${CLAUDE_MD_SECTION_END}
|
|
858
|
-
`;
|
|
859
|
-
content = `${content.trim()}
|
|
860
|
-
|
|
861
|
-
${injectionContent.trim()}
|
|
862
|
-
`;
|
|
863
|
-
fs.writeFileSync(claudeMdPath, content, "utf-8");
|
|
864
|
-
return {
|
|
865
|
-
synced: injection.sourcePostmortems.length,
|
|
866
|
-
claudeMdPath
|
|
867
|
-
};
|
|
868
721
|
}
|
|
869
|
-
|
|
870
|
-
|
|
871
|
-
|
|
872
|
-
|
|
873
|
-
|
|
874
|
-
|
|
875
|
-
|
|
876
|
-
const severityOrder = {
|
|
877
|
-
critical: 0,
|
|
878
|
-
high: 1,
|
|
879
|
-
medium: 2,
|
|
880
|
-
low: 3
|
|
722
|
+
try {
|
|
723
|
+
const stat = await fs.stat(absolutePath);
|
|
724
|
+
if (stat.isFile() && absolutePath.endsWith(".md")) {
|
|
725
|
+
return {
|
|
726
|
+
type: "skill",
|
|
727
|
+
confidence: "high",
|
|
728
|
+
reason: "Single markdown file"
|
|
881
729
|
};
|
|
882
|
-
const minSeverityOrder = severityOrder[this.config.minSyncSeverity];
|
|
883
|
-
for (const meta of index.reports) {
|
|
884
|
-
if (severityOrder[meta.severity] <= minSeverityOrder && meta.status === "active") {
|
|
885
|
-
const report = this.getReport(meta.id);
|
|
886
|
-
if (report) {
|
|
887
|
-
reports.push(report);
|
|
888
|
-
}
|
|
889
|
-
}
|
|
890
|
-
if (reports.length >= this.config.maxSyncItems) {
|
|
891
|
-
break;
|
|
892
|
-
}
|
|
893
|
-
}
|
|
894
730
|
}
|
|
895
|
-
|
|
896
|
-
|
|
897
|
-
|
|
898
|
-
|
|
899
|
-
|
|
900
|
-
|
|
901
|
-
|
|
902
|
-
|
|
903
|
-
|
|
904
|
-
|
|
905
|
-
|
|
906
|
-
|
|
907
|
-
|
|
908
|
-
|
|
909
|
-
|
|
910
|
-
|
|
911
|
-
|
|
912
|
-
|
|
913
|
-
|
|
914
|
-
|
|
915
|
-
|
|
916
|
-
|
|
917
|
-
|
|
918
|
-
|
|
919
|
-
|
|
920
|
-
|
|
921
|
-
|
|
922
|
-
|
|
923
|
-
|
|
924
|
-
|
|
925
|
-
|
|
926
|
-
|
|
927
|
-
|
|
928
|
-
|
|
929
|
-
|
|
930
|
-
|
|
931
|
-
|
|
932
|
-
for (const r of reports.slice(0, 5)) {
|
|
933
|
-
for (const d of r.aiDirectives.slice(0, 2)) {
|
|
934
|
-
allDirectives.add(d);
|
|
935
|
-
}
|
|
936
|
-
}
|
|
937
|
-
for (const d of allDirectives) {
|
|
938
|
-
lines.push(`- ${d}`);
|
|
731
|
+
} catch {
|
|
732
|
+
}
|
|
733
|
+
return dirResult.confidence !== "low" ? dirResult : { type: "skill", confidence: "low", reason: "Default type" };
|
|
734
|
+
}
|
|
735
|
+
function detectFromRepoName(name) {
|
|
736
|
+
const lowerName = name.toLowerCase();
|
|
737
|
+
if (lowerName.includes("mcp-server") || lowerName.includes("mcp_server") || lowerName.startsWith("mcp-") || lowerName.endsWith("-mcp")) {
|
|
738
|
+
return { type: "mcp", confidence: "high", reason: `Name contains MCP pattern: ${name}` };
|
|
739
|
+
}
|
|
740
|
+
if (lowerName.includes("agent") || lowerName.includes("-agent") || lowerName.endsWith("-agent")) {
|
|
741
|
+
return { type: "agent", confidence: "medium", reason: `Name contains agent pattern: ${name}` };
|
|
742
|
+
}
|
|
743
|
+
if (lowerName.includes("hook") || lowerName.includes("-hook") || lowerName.endsWith("-hook")) {
|
|
744
|
+
return { type: "hook", confidence: "medium", reason: `Name contains hook pattern: ${name}` };
|
|
745
|
+
}
|
|
746
|
+
if (lowerName.includes("skill") || lowerName.includes("-skill") || lowerName.endsWith("-skill")) {
|
|
747
|
+
return { type: "skill", confidence: "medium", reason: `Name contains skill pattern: ${name}` };
|
|
748
|
+
}
|
|
749
|
+
return { type: "skill", confidence: "low", reason: "No pattern matched" };
|
|
750
|
+
}
|
|
751
|
+
function detectFromPackageName(packageName) {
|
|
752
|
+
const lowerName = packageName.toLowerCase();
|
|
753
|
+
if (lowerName.startsWith("@modelcontextprotocol/")) {
|
|
754
|
+
return { type: "mcp", confidence: "high", reason: "MCP official scope" };
|
|
755
|
+
}
|
|
756
|
+
if (lowerName.includes("mcp-server") || lowerName.includes("mcp_server")) {
|
|
757
|
+
return { type: "mcp", confidence: "high", reason: "MCP server pattern in name" };
|
|
758
|
+
}
|
|
759
|
+
return detectFromRepoName(packageName);
|
|
760
|
+
}
|
|
761
|
+
function detectFromPackageJson(packageJson) {
|
|
762
|
+
if (packageJson.ccjk && typeof packageJson.ccjk === "object") {
|
|
763
|
+
const ccjk = packageJson.ccjk;
|
|
764
|
+
if (ccjk.type && typeof ccjk.type === "string") {
|
|
765
|
+
const type = ccjk.type;
|
|
766
|
+
if (["skill", "mcp", "agent", "hook"].includes(type)) {
|
|
767
|
+
return { type, confidence: "high", reason: "Explicit ccjk.type field" };
|
|
939
768
|
}
|
|
940
|
-
lines.push("");
|
|
941
|
-
lines.push(`> \u8BE6\u7EC6\u4FE1\u606F\u8BF7\u67E5\u770B \`${this.config.directory}/\` \u76EE\u5F55`);
|
|
942
769
|
}
|
|
943
|
-
return {
|
|
944
|
-
sectionId: "postmortem-warnings",
|
|
945
|
-
title: "\u5DF2\u77E5\u95EE\u9898\u9884\u8B66",
|
|
946
|
-
content: lines.join("\n"),
|
|
947
|
-
priority: 100,
|
|
948
|
-
sourcePostmortems: reports.map((r) => r.id),
|
|
949
|
-
lastUpdated: (/* @__PURE__ */ new Date()).toISOString()
|
|
950
|
-
};
|
|
951
770
|
}
|
|
952
|
-
|
|
953
|
-
|
|
954
|
-
|
|
955
|
-
|
|
956
|
-
|
|
957
|
-
|
|
958
|
-
|
|
959
|
-
|
|
960
|
-
|
|
961
|
-
|
|
962
|
-
|
|
963
|
-
|
|
964
|
-
|
|
965
|
-
|
|
966
|
-
|
|
967
|
-
}
|
|
968
|
-
const issues = [];
|
|
969
|
-
const index = this.loadIndex();
|
|
970
|
-
if (!index) {
|
|
971
|
-
return this.createEmptyCheckReport(filesToCheck.length);
|
|
972
|
-
}
|
|
973
|
-
const patterns = [];
|
|
974
|
-
for (const meta of index.reports) {
|
|
975
|
-
if (meta.status !== "active")
|
|
976
|
-
continue;
|
|
977
|
-
const report = this.getReport(meta.id);
|
|
978
|
-
if (!report)
|
|
979
|
-
continue;
|
|
980
|
-
for (const pattern of report.detectionPatterns) {
|
|
981
|
-
patterns.push({ pattern, postmortemId: report.id });
|
|
771
|
+
if (Array.isArray(packageJson.keywords)) {
|
|
772
|
+
const keywords = packageJson.keywords;
|
|
773
|
+
const keywordMap = {
|
|
774
|
+
"mcp-server": "mcp",
|
|
775
|
+
"mcp": "mcp",
|
|
776
|
+
"model-context-protocol": "mcp",
|
|
777
|
+
"ccjk-skill": "skill",
|
|
778
|
+
"claude-skill": "skill",
|
|
779
|
+
"ccjk-agent": "agent",
|
|
780
|
+
"ccjk-hook": "hook"
|
|
781
|
+
};
|
|
782
|
+
for (const keyword of keywords) {
|
|
783
|
+
const type = keywordMap[keyword.toLowerCase()];
|
|
784
|
+
if (type) {
|
|
785
|
+
return { type, confidence: "high", reason: `Keyword: ${keyword}` };
|
|
982
786
|
}
|
|
983
787
|
}
|
|
984
|
-
|
|
985
|
-
|
|
986
|
-
|
|
987
|
-
|
|
988
|
-
|
|
989
|
-
|
|
990
|
-
for (const { pattern, postmortemId } of patterns) {
|
|
991
|
-
if (!pattern.fileTypes.some((ft) => file.endsWith(ft))) {
|
|
992
|
-
continue;
|
|
993
|
-
}
|
|
994
|
-
if (pattern.type === "regex") {
|
|
995
|
-
try {
|
|
996
|
-
const regex = new RegExp(pattern.pattern, "g");
|
|
997
|
-
for (let i = 0; i < lines.length; i++) {
|
|
998
|
-
const line = lines[i];
|
|
999
|
-
const matches = line.match(regex);
|
|
1000
|
-
if (matches) {
|
|
1001
|
-
issues.push({
|
|
1002
|
-
file,
|
|
1003
|
-
line: i + 1,
|
|
1004
|
-
column: line.indexOf(matches[0]) + 1,
|
|
1005
|
-
pattern,
|
|
1006
|
-
postmortemId,
|
|
1007
|
-
message: `\u53EF\u80FD\u89E6\u53D1 ${postmortemId}: ${pattern.description}`,
|
|
1008
|
-
suggestion: `\u53C2\u8003 ${this.config.directory}/${postmortemId}.md`
|
|
1009
|
-
});
|
|
1010
|
-
}
|
|
1011
|
-
}
|
|
1012
|
-
} catch {
|
|
1013
|
-
}
|
|
1014
|
-
}
|
|
788
|
+
}
|
|
789
|
+
if (packageJson.bin) {
|
|
790
|
+
const binKeys = Object.keys(packageJson.bin);
|
|
791
|
+
for (const key of binKeys) {
|
|
792
|
+
if (key.includes("mcp") || key.includes("server")) {
|
|
793
|
+
return { type: "mcp", confidence: "medium", reason: `Binary name: ${key}` };
|
|
1015
794
|
}
|
|
1016
795
|
}
|
|
1017
|
-
const summary = {
|
|
1018
|
-
critical: issues.filter((i) => i.pattern.severity === "critical").length,
|
|
1019
|
-
high: issues.filter((i) => i.pattern.severity === "high").length,
|
|
1020
|
-
medium: issues.filter((i) => i.pattern.severity === "medium").length,
|
|
1021
|
-
low: issues.filter((i) => i.pattern.severity === "low").length
|
|
1022
|
-
};
|
|
1023
|
-
return {
|
|
1024
|
-
timestamp: (/* @__PURE__ */ new Date()).toISOString(),
|
|
1025
|
-
filesChecked: filesToCheck.length,
|
|
1026
|
-
issuesFound: issues,
|
|
1027
|
-
summary,
|
|
1028
|
-
passed: summary.critical === 0 && summary.high === 0
|
|
1029
|
-
};
|
|
1030
796
|
}
|
|
1031
|
-
|
|
1032
|
-
|
|
1033
|
-
|
|
1034
|
-
|
|
1035
|
-
try {
|
|
1036
|
-
const output = execSync("git diff --cached --name-only", {
|
|
1037
|
-
cwd: this.projectRoot,
|
|
1038
|
-
encoding: "utf-8"
|
|
1039
|
-
});
|
|
1040
|
-
return output.trim().split("\n").filter(Boolean);
|
|
1041
|
-
} catch {
|
|
1042
|
-
return [];
|
|
797
|
+
if (typeof packageJson.main === "string") {
|
|
798
|
+
const main = packageJson.main.toLowerCase();
|
|
799
|
+
if (main.includes("server") || main.includes("mcp")) {
|
|
800
|
+
return { type: "mcp", confidence: "low", reason: `Main file: ${packageJson.main}` };
|
|
1043
801
|
}
|
|
1044
802
|
}
|
|
1045
|
-
|
|
1046
|
-
|
|
1047
|
-
|
|
1048
|
-
|
|
1049
|
-
|
|
1050
|
-
|
|
1051
|
-
|
|
1052
|
-
|
|
1053
|
-
|
|
1054
|
-
|
|
1055
|
-
|
|
1056
|
-
|
|
1057
|
-
|
|
1058
|
-
|
|
1059
|
-
|
|
1060
|
-
|
|
1061
|
-
|
|
1062
|
-
|
|
1063
|
-
|
|
1064
|
-
}
|
|
1065
|
-
}
|
|
803
|
+
return { type: "skill", confidence: "low", reason: "No specific indicators" };
|
|
804
|
+
}
|
|
805
|
+
|
|
806
|
+
async function addCommand(source, options = {}) {
|
|
807
|
+
const { force = false, dryRun = false, json = false, lang = "en" } = options;
|
|
808
|
+
const i18n = getI18n(lang);
|
|
809
|
+
const spinner = json ? null : ora(i18n.parsing).start();
|
|
810
|
+
let sourceInfo;
|
|
811
|
+
try {
|
|
812
|
+
sourceInfo = parseSource(source);
|
|
813
|
+
spinner?.succeed(i18n.parsed(sourceInfo.type));
|
|
814
|
+
} catch (error) {
|
|
815
|
+
spinner?.fail(i18n.parseFailed);
|
|
816
|
+
const result = {
|
|
817
|
+
success: false,
|
|
818
|
+
source,
|
|
819
|
+
sourceType: "local",
|
|
820
|
+
pluginType: "skill",
|
|
821
|
+
error: error instanceof Error ? error.message : String(error)
|
|
1066
822
|
};
|
|
1067
|
-
|
|
1068
|
-
|
|
1069
|
-
|
|
1070
|
-
|
|
1071
|
-
|
|
1072
|
-
|
|
1073
|
-
|
|
1074
|
-
|
|
1075
|
-
return new RegExp(`^${regexPattern}$`).test(filepath);
|
|
823
|
+
if (json) {
|
|
824
|
+
console.log(JSON.stringify(result, null, 2));
|
|
825
|
+
} else {
|
|
826
|
+
console.error(a.red(`
|
|
827
|
+
${i18n.error}: ${result.error}
|
|
828
|
+
`));
|
|
829
|
+
}
|
|
830
|
+
return result;
|
|
1076
831
|
}
|
|
1077
|
-
|
|
1078
|
-
|
|
1079
|
-
|
|
1080
|
-
|
|
1081
|
-
|
|
1082
|
-
|
|
1083
|
-
|
|
1084
|
-
|
|
1085
|
-
|
|
1086
|
-
|
|
1087
|
-
};
|
|
832
|
+
let pluginType = options.type;
|
|
833
|
+
if (!pluginType) {
|
|
834
|
+
spinner?.start(i18n.detectingType);
|
|
835
|
+
try {
|
|
836
|
+
pluginType = await detectPluginType(sourceInfo);
|
|
837
|
+
spinner?.succeed(i18n.detectedType(pluginType));
|
|
838
|
+
} catch {
|
|
839
|
+
spinner?.info(i18n.defaultType);
|
|
840
|
+
pluginType = "skill";
|
|
841
|
+
}
|
|
1088
842
|
}
|
|
1089
|
-
|
|
1090
|
-
|
|
1091
|
-
|
|
1092
|
-
|
|
1093
|
-
|
|
1094
|
-
|
|
1095
|
-
|
|
1096
|
-
|
|
1097
|
-
|
|
1098
|
-
|
|
1099
|
-
|
|
1100
|
-
|
|
1101
|
-
|
|
1102
|
-
|
|
1103
|
-
|
|
1104
|
-
);
|
|
1105
|
-
const existingIds = this.listReports().map((r) => r.id);
|
|
1106
|
-
const newReports = PostmortemAnalyzer.generatePostmortem(analyses, existingIds);
|
|
1107
|
-
const newIds = [];
|
|
1108
|
-
for (const report of newReports) {
|
|
1109
|
-
report.affectedVersions = { from: since || "unknown", to: version };
|
|
1110
|
-
this.saveReport(report);
|
|
1111
|
-
newIds.push(report.id);
|
|
843
|
+
spinner?.start(dryRun ? i18n.previewInstall : i18n.installing);
|
|
844
|
+
try {
|
|
845
|
+
let result;
|
|
846
|
+
switch (sourceInfo.type) {
|
|
847
|
+
case "github":
|
|
848
|
+
result = await installFromGitHub(sourceInfo, pluginType, { force, dryRun });
|
|
849
|
+
break;
|
|
850
|
+
case "npm":
|
|
851
|
+
result = await installFromNpm(sourceInfo, pluginType, { force, dryRun });
|
|
852
|
+
break;
|
|
853
|
+
case "local":
|
|
854
|
+
result = await installFromLocal(sourceInfo, pluginType, { force, dryRun });
|
|
855
|
+
break;
|
|
856
|
+
default:
|
|
857
|
+
throw new Error(`Unsupported source type: ${sourceInfo.type}`);
|
|
1112
858
|
}
|
|
1113
|
-
|
|
1114
|
-
|
|
1115
|
-
|
|
1116
|
-
|
|
1117
|
-
|
|
1118
|
-
|
|
1119
|
-
|
|
1120
|
-
|
|
1121
|
-
|
|
859
|
+
if (result.success) {
|
|
860
|
+
spinner?.succeed(dryRun ? i18n.previewSuccess : i18n.installSuccess);
|
|
861
|
+
} else {
|
|
862
|
+
spinner?.fail(i18n.installFailed);
|
|
863
|
+
}
|
|
864
|
+
if (json) {
|
|
865
|
+
console.log(JSON.stringify(result, null, 2));
|
|
866
|
+
} else if (result.success && !dryRun) {
|
|
867
|
+
printSuccessMessage(result, i18n);
|
|
868
|
+
} else if (result.success && dryRun) {
|
|
869
|
+
printPreviewMessage(result, i18n);
|
|
870
|
+
} else {
|
|
871
|
+
console.error(a.red(`
|
|
872
|
+
${i18n.error}: ${result.error}
|
|
873
|
+
`));
|
|
874
|
+
}
|
|
875
|
+
return result;
|
|
876
|
+
} catch (error) {
|
|
877
|
+
spinner?.fail(i18n.installFailed);
|
|
878
|
+
const result = {
|
|
879
|
+
success: false,
|
|
880
|
+
source,
|
|
881
|
+
sourceType: sourceInfo.type,
|
|
882
|
+
pluginType,
|
|
883
|
+
error: error instanceof Error ? error.message : String(error)
|
|
1122
884
|
};
|
|
1123
|
-
|
|
1124
|
-
|
|
1125
|
-
|
|
1126
|
-
|
|
885
|
+
if (json) {
|
|
886
|
+
console.log(JSON.stringify(result, null, 2));
|
|
887
|
+
} else {
|
|
888
|
+
console.error(a.red(`
|
|
889
|
+
${i18n.error}: ${result.error}
|
|
890
|
+
`));
|
|
1127
891
|
}
|
|
1128
|
-
return
|
|
892
|
+
return result;
|
|
1129
893
|
}
|
|
1130
|
-
|
|
1131
|
-
|
|
1132
|
-
|
|
1133
|
-
|
|
1134
|
-
|
|
1135
|
-
|
|
1136
|
-
|
|
1137
|
-
|
|
1138
|
-
|
|
1139
|
-
|
|
1140
|
-
|
|
1141
|
-
|
|
1142
|
-
|
|
1143
|
-
|
|
1144
|
-
|
|
1145
|
-
|
|
1146
|
-
|
|
894
|
+
}
|
|
895
|
+
function printSuccessMessage(result, i18n) {
|
|
896
|
+
console.log();
|
|
897
|
+
console.log(a.green.bold(`\u2713 ${i18n.installed}`));
|
|
898
|
+
console.log();
|
|
899
|
+
console.log(` ${a.gray(i18n.name)}: ${result.details?.name || result.source}`);
|
|
900
|
+
if (result.details?.version) {
|
|
901
|
+
console.log(` ${a.gray(i18n.version)}: ${result.details.version}`);
|
|
902
|
+
}
|
|
903
|
+
console.log(` ${a.gray(i18n.type)}: ${result.pluginType}`);
|
|
904
|
+
console.log(` ${a.gray(i18n.path)}: ${result.installedPath}`);
|
|
905
|
+
console.log();
|
|
906
|
+
console.log(a.cyan(i18n.nextSteps));
|
|
907
|
+
switch (result.pluginType) {
|
|
908
|
+
case "skill":
|
|
909
|
+
console.log(` ${a.gray("\u2022")} ${i18n.skillHint}`);
|
|
910
|
+
break;
|
|
911
|
+
case "mcp":
|
|
912
|
+
console.log(` ${a.gray("\u2022")} ${i18n.mcpHint}`);
|
|
913
|
+
break;
|
|
914
|
+
case "agent":
|
|
915
|
+
console.log(` ${a.gray("\u2022")} ${i18n.agentHint}`);
|
|
916
|
+
break;
|
|
917
|
+
case "hook":
|
|
918
|
+
console.log(` ${a.gray("\u2022")} ${i18n.hookHint}`);
|
|
919
|
+
break;
|
|
920
|
+
}
|
|
921
|
+
console.log();
|
|
922
|
+
}
|
|
923
|
+
function printPreviewMessage(result, i18n) {
|
|
924
|
+
console.log();
|
|
925
|
+
console.log(a.yellow.bold(`\u26A1 ${i18n.preview}`));
|
|
926
|
+
console.log();
|
|
927
|
+
console.log(` ${a.gray(i18n.source)}: ${result.source}`);
|
|
928
|
+
console.log(` ${a.gray(i18n.type)}: ${result.pluginType}`);
|
|
929
|
+
console.log(` ${a.gray(i18n.target)}: ${result.installedPath}`);
|
|
930
|
+
if (result.details?.files?.length) {
|
|
931
|
+
console.log(` ${a.gray(i18n.files)}:`);
|
|
932
|
+
for (const file of result.details.files.slice(0, 5)) {
|
|
933
|
+
console.log(` ${a.gray("\u2022")} ${file}`);
|
|
1147
934
|
}
|
|
1148
|
-
|
|
1149
|
-
|
|
1150
|
-
/**
|
|
1151
|
-
* 提取关键教训
|
|
1152
|
-
*/
|
|
1153
|
-
extractKeyLessons(reports) {
|
|
1154
|
-
const lessons = /* @__PURE__ */ new Set();
|
|
1155
|
-
for (const report of reports) {
|
|
1156
|
-
for (const measure of report.preventionMeasures.slice(0, 2)) {
|
|
1157
|
-
lessons.add(measure);
|
|
1158
|
-
}
|
|
935
|
+
if (result.details.files.length > 5) {
|
|
936
|
+
console.log(` ${a.gray("...")} ${i18n.andMore(result.details.files.length - 5)}`);
|
|
1159
937
|
}
|
|
1160
|
-
return Array.from(lessons).slice(0, 10);
|
|
1161
938
|
}
|
|
939
|
+
console.log();
|
|
940
|
+
console.log(a.gray(i18n.dryRunNote));
|
|
941
|
+
console.log();
|
|
1162
942
|
}
|
|
1163
|
-
|
|
1164
|
-
|
|
1165
|
-
|
|
1166
|
-
|
|
1167
|
-
|
|
1168
|
-
|
|
943
|
+
function getI18n(lang) {
|
|
944
|
+
const texts = {
|
|
945
|
+
"zh-CN": {
|
|
946
|
+
parsing: "\u89E3\u6790\u6765\u6E90...",
|
|
947
|
+
parsed: (type) => `\u6765\u6E90\u7C7B\u578B: ${type}`,
|
|
948
|
+
parseFailed: "\u89E3\u6790\u6765\u6E90\u5931\u8D25",
|
|
949
|
+
detectingType: "\u68C0\u6D4B\u63D2\u4EF6\u7C7B\u578B...",
|
|
950
|
+
detectedType: (type) => `\u68C0\u6D4B\u5230\u7C7B\u578B: ${type}`,
|
|
951
|
+
defaultType: "\u4F7F\u7528\u9ED8\u8BA4\u7C7B\u578B: skill",
|
|
952
|
+
installing: "\u5B89\u88C5\u4E2D...",
|
|
953
|
+
previewInstall: "\u9884\u89C8\u5B89\u88C5...",
|
|
954
|
+
installSuccess: "\u5B89\u88C5\u6210\u529F",
|
|
955
|
+
previewSuccess: "\u9884\u89C8\u5B8C\u6210",
|
|
956
|
+
installFailed: "\u5B89\u88C5\u5931\u8D25",
|
|
957
|
+
error: "\u9519\u8BEF",
|
|
958
|
+
installed: "\u63D2\u4EF6\u5DF2\u5B89\u88C5",
|
|
959
|
+
preview: "\u5B89\u88C5\u9884\u89C8 (dry-run)",
|
|
960
|
+
name: "\u540D\u79F0",
|
|
961
|
+
version: "\u7248\u672C",
|
|
962
|
+
type: "\u7C7B\u578B",
|
|
963
|
+
path: "\u8DEF\u5F84",
|
|
964
|
+
source: "\u6765\u6E90",
|
|
965
|
+
target: "\u76EE\u6807",
|
|
966
|
+
files: "\u6587\u4EF6",
|
|
967
|
+
nextSteps: "\u4E0B\u4E00\u6B65:",
|
|
968
|
+
skillHint: "\u4F7F\u7528 /skill-name \u5728 Claude Code \u4E2D\u8C03\u7528",
|
|
969
|
+
mcpHint: "\u8FD0\u884C ccjk mcp status \u67E5\u770B\u72B6\u6001",
|
|
970
|
+
agentHint: "\u4F7F\u7528 ccjk agent list \u67E5\u770B\u5DF2\u5B89\u88C5\u7684 agent",
|
|
971
|
+
hookHint: "\u8FD0\u884C ccjk hooks list \u67E5\u770B\u5DF2\u5B89\u88C5\u7684 hook",
|
|
972
|
+
dryRunNote: "\u8FD9\u662F\u9884\u89C8\u6A21\u5F0F\uFF0C\u672A\u5B9E\u9645\u5B89\u88C5\u3002\u79FB\u9664 --dry-run \u6267\u884C\u5B89\u88C5\u3002",
|
|
973
|
+
andMore: (n) => `\u8FD8\u6709 ${n} \u4E2A\u6587\u4EF6`
|
|
974
|
+
},
|
|
975
|
+
"en": {
|
|
976
|
+
parsing: "Parsing source...",
|
|
977
|
+
parsed: (type) => `Source type: ${type}`,
|
|
978
|
+
parseFailed: "Failed to parse source",
|
|
979
|
+
detectingType: "Detecting plugin type...",
|
|
980
|
+
detectedType: (type) => `Detected type: ${type}`,
|
|
981
|
+
defaultType: "Using default type: skill",
|
|
982
|
+
installing: "Installing...",
|
|
983
|
+
previewInstall: "Previewing installation...",
|
|
984
|
+
installSuccess: "Installation successful",
|
|
985
|
+
previewSuccess: "Preview complete",
|
|
986
|
+
installFailed: "Installation failed",
|
|
987
|
+
error: "Error",
|
|
988
|
+
installed: "Plugin installed",
|
|
989
|
+
preview: "Installation preview (dry-run)",
|
|
990
|
+
name: "Name",
|
|
991
|
+
version: "Version",
|
|
992
|
+
type: "Type",
|
|
993
|
+
path: "Path",
|
|
994
|
+
source: "Source",
|
|
995
|
+
target: "Target",
|
|
996
|
+
files: "Files",
|
|
997
|
+
nextSteps: "Next steps:",
|
|
998
|
+
skillHint: "Use /skill-name in Claude Code to invoke",
|
|
999
|
+
mcpHint: "Run ccjk mcp status to check status",
|
|
1000
|
+
agentHint: "Use ccjk agent list to see installed agents",
|
|
1001
|
+
hookHint: "Run ccjk hooks list to see installed hooks",
|
|
1002
|
+
dryRunNote: "This is preview mode, nothing was installed. Remove --dry-run to install.",
|
|
1003
|
+
andMore: (n) => `and ${n} more files`
|
|
1004
|
+
}
|
|
1005
|
+
};
|
|
1006
|
+
return texts[lang];
|
|
1169
1007
|
}
|
|
1170
1008
|
|
|
1171
|
-
export {
|
|
1009
|
+
export { addCommand, parseSource };
|