ccjk 11.1.2 → 12.0.0
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 +1 -1
- package/dist/chunks/agent.mjs +1 -1
- package/dist/chunks/api-cli.mjs +118 -0
- package/dist/chunks/api-providers.mjs +1 -1
- package/dist/chunks/api.mjs +8 -759
- package/dist/chunks/auto-bootstrap.mjs +1 -1
- package/dist/chunks/auto-init.mjs +1 -1
- package/dist/chunks/auto-updater.mjs +1 -1
- package/dist/chunks/banner.mjs +1 -1
- package/dist/chunks/boost.mjs +1 -1
- package/dist/chunks/ccjk-agents.mjs +1 -1
- package/dist/chunks/ccjk-all.mjs +1 -1
- package/dist/chunks/ccjk-config.mjs +1 -1
- package/dist/chunks/ccjk-hooks.mjs +1 -1
- package/dist/chunks/ccjk-mcp.mjs +1 -1
- package/dist/chunks/ccjk-setup.mjs +1 -1
- package/dist/chunks/ccjk-skills.mjs +1 -1
- package/dist/chunks/ccr.mjs +6 -6
- package/dist/chunks/ccu.mjs +1 -1
- package/dist/chunks/check-updates.mjs +2 -2
- package/dist/chunks/claude-code-config-manager.mjs +2 -2
- package/dist/chunks/claude-code-incremental-manager.mjs +2 -2
- package/dist/chunks/claude-config.mjs +1 -1
- package/dist/chunks/claude-wrapper.mjs +1 -1
- package/dist/chunks/codex-config-switch.mjs +1 -1
- package/dist/chunks/codex-provider-manager.mjs +1 -1
- package/dist/chunks/codex-uninstaller.mjs +1 -1
- package/dist/chunks/codex.mjs +1 -1
- package/dist/chunks/commands.mjs +1 -1
- package/dist/chunks/completion.mjs +1 -1
- package/dist/chunks/config-consolidator.mjs +1 -1
- package/dist/chunks/config-switch.mjs +2 -2
- package/dist/chunks/config.mjs +1 -1
- package/dist/chunks/config2.mjs +1 -1
- package/dist/chunks/config3.mjs +1 -1
- package/dist/chunks/constants.mjs +1 -1
- package/dist/chunks/dashboard.mjs +1 -1
- package/dist/chunks/doctor.mjs +31 -7
- package/dist/chunks/evolution.mjs +1 -1
- package/dist/chunks/features.mjs +5 -5
- package/dist/chunks/index.mjs +7 -7
- package/dist/chunks/index2.mjs +10 -177
- package/dist/chunks/index3.mjs +168 -1162
- package/dist/chunks/index4.mjs +1076 -910
- package/dist/chunks/index5.mjs +947 -137
- package/dist/chunks/index6.mjs +167 -635
- package/dist/chunks/index7.mjs +663 -0
- package/dist/chunks/init.mjs +17 -17
- package/dist/chunks/installer.mjs +1 -1
- package/dist/chunks/interview.mjs +2 -2
- package/dist/chunks/marketplace.mjs +1 -1
- package/dist/chunks/mcp-cli.mjs +191 -0
- package/dist/chunks/mcp.mjs +2 -2
- package/dist/chunks/menu.mjs +3 -3
- package/dist/chunks/migrator.mjs +1 -1
- package/dist/chunks/monitor.mjs +1 -1
- package/dist/chunks/notification.mjs +1 -1
- package/dist/chunks/onboarding.mjs +1 -1
- package/dist/chunks/package.mjs +1 -1
- package/dist/chunks/permission-manager.mjs +1 -1
- package/dist/chunks/permissions.mjs +1 -1
- package/dist/chunks/persistence-manager.mjs +1 -1
- package/dist/chunks/plugin.mjs +1 -1
- package/dist/chunks/prompts.mjs +1 -1
- package/dist/chunks/providers.mjs +1 -1
- package/dist/chunks/quick-actions.mjs +1 -1
- package/dist/chunks/quick-setup.mjs +5 -5
- package/dist/chunks/remote.mjs +2 -2
- package/dist/chunks/silent-updater.mjs +1 -1
- package/dist/chunks/simple-config.mjs +1 -1
- package/dist/chunks/skill.mjs +1 -1
- package/dist/chunks/skills-sync.mjs +1 -1
- package/dist/chunks/skills.mjs +1 -1
- package/dist/chunks/slash-commands.mjs +2 -2
- package/dist/chunks/startup.mjs +1 -1
- package/dist/chunks/stats.mjs +1 -1
- package/dist/chunks/status.mjs +1 -1
- package/dist/chunks/team.mjs +1 -1
- package/dist/chunks/thinking.mjs +2 -2
- package/dist/chunks/uninstall.mjs +3 -3
- package/dist/chunks/update.mjs +4 -4
- package/dist/chunks/upgrade-manager.mjs +1 -1
- package/dist/chunks/version-checker.mjs +1 -1
- package/dist/chunks/vim.mjs +1 -1
- package/dist/chunks/zero-config.mjs +1 -1
- package/dist/cli.mjs +76 -16
- package/dist/index.d.mts +37 -10
- package/dist/index.d.ts +37 -10
- package/dist/index.mjs +2 -2
- package/dist/shared/{ccjk.DB2UYcq0.mjs → ccjk.BOIUTf5z.mjs} +2 -2
- package/dist/shared/ccjk.CL4Yat0G.mjs +303 -0
- package/dist/shared/{ccjk.CxtuJxaS.mjs → ccjk.CN0edl87.mjs} +1 -1
- package/dist/shared/{ccjk.BKoi8-Hy.mjs → ccjk.CrB6OYHv.mjs} +1 -1
- package/dist/shared/{ccjk.DVBW2wxp.mjs → ccjk.Dk1HDseQ.mjs} +1 -1
- package/dist/shared/ccjk.bhFAMRyc.mjs +460 -0
- package/dist/shared/ccjk.cChAaGgT.mjs +88 -0
- package/dist/shared/{ccjk.DrMygfCF.mjs → ccjk.j4uut26D.mjs} +1 -1
- package/package.json +2 -3
- package/dist/shared/ccjk.Cu_R2MbQ.mjs +0 -75
package/dist/chunks/index3.mjs
CHANGED
|
@@ -1,1171 +1,177 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import
|
|
3
|
-
import
|
|
4
|
-
import
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
}
|
|
92
|
-
function detectBugType(message, diff) {
|
|
93
|
-
const content = `${message}
|
|
94
|
-
${diff}`.toLowerCase();
|
|
95
|
-
const patterns = [
|
|
96
|
-
{
|
|
97
|
-
category: "type-safety",
|
|
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
|
-
]
|
|
164
|
-
}
|
|
165
|
-
];
|
|
166
|
-
for (const { category, patterns: categoryPatterns } of patterns) {
|
|
167
|
-
if (categoryPatterns.some((p) => p.test(content))) {
|
|
168
|
-
return category;
|
|
169
|
-
}
|
|
170
|
-
}
|
|
171
|
-
return "other";
|
|
172
|
-
}
|
|
173
|
-
function detectSeverity(message, diff, files) {
|
|
174
|
-
const content = `${message}
|
|
175
|
-
${diff}`.toLowerCase();
|
|
176
|
-
if (/critical|crash|data.*loss|security.*vuln/i.test(content) || files.some((f) => /auth|security|payment/i.test(f))) {
|
|
177
|
-
return "critical";
|
|
178
|
-
}
|
|
179
|
-
if (/breaking|major|important|urgent/i.test(content) || files.length > 10) {
|
|
180
|
-
return "high";
|
|
181
|
-
}
|
|
182
|
-
if (/moderate|minor.*issue/i.test(content) || files.length > 3) {
|
|
183
|
-
return "medium";
|
|
184
|
-
}
|
|
185
|
-
return "low";
|
|
186
|
-
}
|
|
187
|
-
function extractRootCause(message, diff) {
|
|
188
|
-
const causePatterns = [
|
|
189
|
-
/caused by[:\s]+(.+)/i,
|
|
190
|
-
/root cause[:\s]+(.+)/i,
|
|
191
|
-
/because[:\s]+(.+)/i,
|
|
192
|
-
/due to[:\s]+(.+)/i,
|
|
193
|
-
/原因[::]\s*(.+)/,
|
|
194
|
-
/由于\s*(.+)/
|
|
195
|
-
];
|
|
196
|
-
for (const pattern of causePatterns) {
|
|
197
|
-
const match = message.match(pattern);
|
|
198
|
-
if (match) {
|
|
199
|
-
return match[1].trim();
|
|
200
|
-
}
|
|
201
|
-
}
|
|
202
|
-
const removedLines = diff.match(/^-[^-].*/gm) || [];
|
|
203
|
-
const addedLines = diff.match(/^\+[^+].*/gm) || [];
|
|
204
|
-
if (removedLines.length > 0 && addedLines.length > 0) {
|
|
205
|
-
return `\u4EE3\u7801\u53D8\u66F4: \u79FB\u9664 ${removedLines.length} \u884C, \u65B0\u589E ${addedLines.length} \u884C`;
|
|
206
|
-
}
|
|
207
|
-
return "\u9700\u8981\u8FDB\u4E00\u6B65\u5206\u6790";
|
|
208
|
-
}
|
|
209
|
-
function extractSolution(diff) {
|
|
210
|
-
const addedLines = diff.match(/^\+[^+].*/gm) || [];
|
|
211
|
-
if (addedLines.length === 0) {
|
|
212
|
-
return "\u5220\u9664\u4E86\u95EE\u9898\u4EE3\u7801";
|
|
213
|
-
}
|
|
214
|
-
if (addedLines.length <= 5) {
|
|
215
|
-
return addedLines.map((l) => l.substring(1)).join("\n");
|
|
216
|
-
}
|
|
217
|
-
return `\u65B0\u589E ${addedLines.length} \u884C\u4EE3\u7801\u4FEE\u590D\u95EE\u9898`;
|
|
218
|
-
}
|
|
219
|
-
function generatePreventionSuggestions(bugType, _diff) {
|
|
220
|
-
const suggestions = {
|
|
221
|
-
"type-safety": [
|
|
222
|
-
"\u542F\u7528 TypeScript strict \u6A21\u5F0F",
|
|
223
|
-
"\u4F7F\u7528\u53EF\u9009\u94FE\u64CD\u4F5C\u7B26 (?.) \u8FDB\u884C\u7A7A\u503C\u68C0\u67E5",
|
|
224
|
-
"\u4E3A\u6240\u6709\u5916\u90E8\u6570\u636E\u6DFB\u52A0\u8FD0\u884C\u65F6\u9A8C\u8BC1",
|
|
225
|
-
"\u4F7F\u7528 zod \u6216 io-ts \u8FDB\u884C\u7C7B\u578B\u9A8C\u8BC1"
|
|
226
|
-
],
|
|
227
|
-
"error-handling": [
|
|
228
|
-
"\u4E3A\u6240\u6709\u5F02\u6B65\u64CD\u4F5C\u6DFB\u52A0 try-catch",
|
|
229
|
-
"\u5B9E\u73B0\u5168\u5C40\u9519\u8BEF\u5904\u7406\u4E2D\u95F4\u4EF6",
|
|
230
|
-
"\u6DFB\u52A0\u9519\u8BEF\u8FB9\u754C\u7EC4\u4EF6 (React)",
|
|
231
|
-
"\u4F7F\u7528 Result \u7C7B\u578B\u66FF\u4EE3\u5F02\u5E38"
|
|
232
|
-
],
|
|
233
|
-
"performance": [
|
|
234
|
-
"\u6DFB\u52A0\u6027\u80FD\u76D1\u63A7\u548C\u544A\u8B66",
|
|
235
|
-
"\u5B9E\u73B0\u7F13\u5B58\u7B56\u7565",
|
|
236
|
-
"\u4F7F\u7528\u61D2\u52A0\u8F7D\u548C\u4EE3\u7801\u5206\u5272",
|
|
237
|
-
"\u5B9A\u671F\u8FDB\u884C\u6027\u80FD\u6D4B\u8BD5"
|
|
238
|
-
],
|
|
239
|
-
"security": [
|
|
240
|
-
"\u5B9E\u65BD\u5B89\u5168\u4EE3\u7801\u5BA1\u67E5",
|
|
241
|
-
"\u4F7F\u7528\u5B89\u5168\u626B\u63CF\u5DE5\u5177",
|
|
242
|
-
"\u9075\u5FAA OWASP \u5B89\u5168\u6307\u5357",
|
|
243
|
-
"\u5B9A\u671F\u66F4\u65B0\u4F9D\u8D56"
|
|
244
|
-
],
|
|
245
|
-
"race-condition": [
|
|
246
|
-
"\u4F7F\u7528\u9002\u5F53\u7684\u9501\u673A\u5236",
|
|
247
|
-
"\u907F\u514D\u5171\u4EAB\u53EF\u53D8\u72B6\u6001",
|
|
248
|
-
"\u4F7F\u7528\u539F\u5B50\u64CD\u4F5C",
|
|
249
|
-
"\u6DFB\u52A0\u5E76\u53D1\u6D4B\u8BD5"
|
|
250
|
-
],
|
|
251
|
-
"logic-error": [
|
|
252
|
-
"\u589E\u52A0\u5355\u5143\u6D4B\u8BD5\u8986\u76D6",
|
|
253
|
-
"\u5B9E\u65BD\u4EE3\u7801\u5BA1\u67E5",
|
|
254
|
-
"\u6DFB\u52A0\u65AD\u8A00\u548C\u4E0D\u53D8\u91CF\u68C0\u67E5",
|
|
255
|
-
"\u4F7F\u7528\u5F62\u5F0F\u5316\u9A8C\u8BC1\u5DE5\u5177"
|
|
256
|
-
],
|
|
257
|
-
"api-misuse": [
|
|
258
|
-
"\u5B8C\u5584 API \u6587\u6863",
|
|
259
|
-
"\u6DFB\u52A0\u53C2\u6570\u9A8C\u8BC1",
|
|
260
|
-
"\u63D0\u4F9B\u4F7F\u7528\u793A\u4F8B",
|
|
261
|
-
"\u5B9E\u73B0 API \u7248\u672C\u63A7\u5236"
|
|
262
|
-
],
|
|
263
|
-
"configuration": [
|
|
264
|
-
"\u4F7F\u7528\u914D\u7F6E\u9A8C\u8BC1",
|
|
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"
|
|
323
|
-
}
|
|
324
|
-
};
|
|
325
|
-
reports.push(report);
|
|
326
|
-
}
|
|
327
|
-
}
|
|
328
|
-
return reports;
|
|
329
|
-
}
|
|
330
|
-
function groupByCategory(analyses) {
|
|
331
|
-
const grouped = {};
|
|
332
|
-
for (const analysis of analyses) {
|
|
333
|
-
const category = analysis.bugType;
|
|
334
|
-
if (!grouped[category]) {
|
|
335
|
-
grouped[category] = [];
|
|
336
|
-
}
|
|
337
|
-
grouped[category].push(analysis);
|
|
338
|
-
}
|
|
339
|
-
return grouped;
|
|
340
|
-
}
|
|
341
|
-
function mergeAnalyses(analyses) {
|
|
342
|
-
const merged = [];
|
|
343
|
-
for (const analysis of analyses) {
|
|
344
|
-
const similar = merged.find(
|
|
345
|
-
(m) => calculateFileOverlap(m.commit.files, analysis.commit.files) > 0.5
|
|
1
|
+
import { existsSync } from 'node:fs';
|
|
2
|
+
import process__default from 'node:process';
|
|
3
|
+
import { fileURLToPath } from 'node:url';
|
|
4
|
+
import i18next from 'i18next';
|
|
5
|
+
import Backend from 'i18next-fs-backend';
|
|
6
|
+
import { dirname, join } from 'pathe';
|
|
7
|
+
|
|
8
|
+
const i18n = i18next.createInstance();
|
|
9
|
+
const NAMESPACES = [
|
|
10
|
+
"agent-teams",
|
|
11
|
+
// Agent Teams quick-enable
|
|
12
|
+
"agentBrowser",
|
|
13
|
+
// Agent Browser - AI browser automation
|
|
14
|
+
"common",
|
|
15
|
+
"api",
|
|
16
|
+
"ccr",
|
|
17
|
+
"ccjk",
|
|
18
|
+
// CCJK-specific translations
|
|
19
|
+
"ccjk-skills",
|
|
20
|
+
// CCJK Skills installation command
|
|
21
|
+
"ccjk-agents",
|
|
22
|
+
// CCJK Agents management command
|
|
23
|
+
"ccjk-all",
|
|
24
|
+
// CCJK All-in-one setup command
|
|
25
|
+
"cli",
|
|
26
|
+
"cloud-setup",
|
|
27
|
+
// Cloud-based setup wizard
|
|
28
|
+
"cloudPlugins",
|
|
29
|
+
// Cloud-based plugin system
|
|
30
|
+
"cometix",
|
|
31
|
+
"codex",
|
|
32
|
+
"configuration",
|
|
33
|
+
"context",
|
|
34
|
+
// Context compression system
|
|
35
|
+
"errors",
|
|
36
|
+
"hooks",
|
|
37
|
+
// Git hooks management
|
|
38
|
+
"hooksSync",
|
|
39
|
+
// Hooks cloud synchronization
|
|
40
|
+
"installation",
|
|
41
|
+
"interview",
|
|
42
|
+
// Interview-Driven Development
|
|
43
|
+
"language",
|
|
44
|
+
"lsp",
|
|
45
|
+
// Language Server Protocol (v3.8+)
|
|
46
|
+
"marketplace",
|
|
47
|
+
// Marketplace system for plugins, skills, and workflows
|
|
48
|
+
"mcp",
|
|
49
|
+
"menu",
|
|
50
|
+
"multi-config",
|
|
51
|
+
"notification",
|
|
52
|
+
// Task completion notifications
|
|
53
|
+
"permissions",
|
|
54
|
+
// Permission system for API providers, models, and tools
|
|
55
|
+
"plugins",
|
|
56
|
+
// Cloud plugins management
|
|
57
|
+
"registry",
|
|
58
|
+
// Version checking, China detection, and multiple installations
|
|
59
|
+
"sandbox",
|
|
60
|
+
// Sandbox mode for secure request/response handling
|
|
61
|
+
"setup",
|
|
62
|
+
// Setup wizard
|
|
63
|
+
"shencha",
|
|
64
|
+
"skills",
|
|
65
|
+
// Skills management system
|
|
66
|
+
"skillsSync",
|
|
67
|
+
// Skills cloud synchronization
|
|
68
|
+
"smartGuide",
|
|
69
|
+
// Smart Guide for quick actions
|
|
70
|
+
"stats",
|
|
71
|
+
// Usage statistics
|
|
72
|
+
"superpowers",
|
|
73
|
+
// Superpowers plugin integration
|
|
74
|
+
"team",
|
|
75
|
+
"thinking",
|
|
76
|
+
// Thinking mode for Claude Code CLI 2.0.67+
|
|
77
|
+
"tools",
|
|
78
|
+
"uninstall",
|
|
79
|
+
"updater",
|
|
80
|
+
"vim",
|
|
81
|
+
// Vim mode enhancement for Claude Code CLI 2.1.0+
|
|
82
|
+
"workflow",
|
|
83
|
+
"cloud-sync",
|
|
84
|
+
"workspace"
|
|
85
|
+
// Workspace diagnostics and guide
|
|
86
|
+
];
|
|
87
|
+
function ensureI18nInitialized() {
|
|
88
|
+
if (!i18n.isInitialized) {
|
|
89
|
+
throw new Error(
|
|
90
|
+
"i18n is not initialized. Please call initI18n() in CLI command before using utility functions."
|
|
346
91
|
);
|
|
347
|
-
if (similar) {
|
|
348
|
-
similar.commit.files = Array.from(/* @__PURE__ */ new Set([...similar.commit.files, ...analysis.commit.files]));
|
|
349
|
-
similar.preventionSuggestions = Array.from(/* @__PURE__ */ new Set([...similar.preventionSuggestions, ...analysis.preventionSuggestions]));
|
|
350
|
-
} else {
|
|
351
|
-
merged.push({ ...analysis });
|
|
352
|
-
}
|
|
353
92
|
}
|
|
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
|
-
}
|
|
363
|
-
function getNextId(existingIds) {
|
|
364
|
-
const numbers = existingIds.map((id) => Number.parseInt(id.replace("PM-", ""), 10)).filter((n) => !Number.isNaN(n));
|
|
365
|
-
return numbers.length > 0 ? Math.max(...numbers) + 1 : 1;
|
|
366
|
-
}
|
|
367
|
-
function generateTitle(analysis) {
|
|
368
|
-
const categoryTitles = {
|
|
369
|
-
"type-safety": "\u7C7B\u578B\u5B89\u5168\u95EE\u9898",
|
|
370
|
-
"error-handling": "\u9519\u8BEF\u5904\u7406\u7F3A\u5931",
|
|
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}`;
|
|
386
|
-
}
|
|
387
|
-
return baseTitle;
|
|
388
|
-
}
|
|
389
|
-
function generateDescription(analysis) {
|
|
390
|
-
return `
|
|
391
|
-
\u5728 ${analysis.commit.date} \u53D1\u73B0\u5E76\u4FEE\u590D\u4E86\u4E00\u4E2A ${analysis.bugType} \u7C7B\u578B\u7684\u95EE\u9898\u3002
|
|
392
|
-
|
|
393
|
-
**\u63D0\u4EA4\u4FE1\u606F**: ${analysis.commit.message}
|
|
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();
|
|
400
93
|
}
|
|
401
|
-
function
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
|
|
425
|
-
|
|
426
|
-
|
|
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`);
|
|
463
|
-
}
|
|
464
|
-
if (file.includes("config")) {
|
|
465
|
-
directives.push(`\u4FEE\u6539\u914D\u7F6E\u6587\u4EF6\u65F6\u786E\u4FDD\u5411\u540E\u517C\u5BB9`);
|
|
466
|
-
}
|
|
467
|
-
}
|
|
468
|
-
return Array.from(new Set(directives));
|
|
469
|
-
}
|
|
470
|
-
function generateDetectionPatterns(analysis) {
|
|
471
|
-
const patterns = [];
|
|
472
|
-
const categoryPatterns = {
|
|
473
|
-
"type-safety": [
|
|
474
|
-
{
|
|
475
|
-
type: "regex",
|
|
476
|
-
pattern: "\\.\\w+\\.\\w+\\.\\w+(?!\\?)",
|
|
477
|
-
description: "\u8FDE\u7EED\u5C5E\u6027\u8BBF\u95EE\u672A\u4F7F\u7528\u53EF\u9009\u94FE",
|
|
478
|
-
fileTypes: [".ts", ".tsx", ".js", ".jsx"],
|
|
479
|
-
severity: "medium"
|
|
480
|
-
},
|
|
481
|
-
{
|
|
482
|
-
type: "regex",
|
|
483
|
-
pattern: "as any",
|
|
484
|
-
description: "\u4F7F\u7528 any \u7C7B\u578B\u65AD\u8A00",
|
|
485
|
-
fileTypes: [".ts", ".tsx"],
|
|
486
|
-
severity: "low"
|
|
487
|
-
}
|
|
488
|
-
],
|
|
489
|
-
"error-handling": [
|
|
490
|
-
{
|
|
491
|
-
type: "regex",
|
|
492
|
-
pattern: "catch\\s*\\(\\s*\\w*\\s*\\)\\s*\\{\\s*\\}",
|
|
493
|
-
description: "\u7A7A\u7684 catch \u5757",
|
|
494
|
-
fileTypes: [".ts", ".tsx", ".js", ".jsx"],
|
|
495
|
-
severity: "high"
|
|
496
|
-
},
|
|
497
|
-
{
|
|
498
|
-
type: "regex",
|
|
499
|
-
pattern: "\\.then\\([^)]+\\)(?!\\.catch)",
|
|
500
|
-
description: "Promise \u672A\u5904\u7406 rejection",
|
|
501
|
-
fileTypes: [".ts", ".tsx", ".js", ".jsx"],
|
|
502
|
-
severity: "medium"
|
|
503
|
-
}
|
|
504
|
-
],
|
|
505
|
-
"performance": [
|
|
506
|
-
{
|
|
507
|
-
type: "regex",
|
|
508
|
-
pattern: "for\\s*\\([^)]+\\)\\s*\\{[^}]*await",
|
|
509
|
-
description: "\u5FAA\u73AF\u4E2D\u4F7F\u7528 await",
|
|
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"
|
|
542
|
-
}
|
|
543
|
-
],
|
|
544
|
-
"other": []
|
|
545
|
-
};
|
|
546
|
-
patterns.push(...categoryPatterns[analysis.bugType] || []);
|
|
547
|
-
return patterns;
|
|
548
|
-
}
|
|
549
|
-
const PostmortemAnalyzer = {
|
|
550
|
-
getFixCommits,
|
|
551
|
-
analyzeFixCommit,
|
|
552
|
-
generatePostmortem
|
|
553
|
-
};
|
|
554
|
-
|
|
555
|
-
const DEFAULT_CONFIG = {
|
|
556
|
-
enabled: true,
|
|
557
|
-
directory: "./postmortem",
|
|
558
|
-
autoSyncToClaudeMd: true,
|
|
559
|
-
maxSyncItems: 10,
|
|
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"
|
|
568
|
-
}
|
|
569
|
-
};
|
|
570
|
-
const INDEX_FILE = "index.json";
|
|
571
|
-
const CLAUDE_MD_SECTION_START = "<!-- POSTMORTEM_START -->";
|
|
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);
|
|
581
|
-
}
|
|
582
|
-
// ==========================================================================
|
|
583
|
-
// Initialization
|
|
584
|
-
// ==========================================================================
|
|
585
|
-
/**
|
|
586
|
-
* 初始化 Postmortem 系统
|
|
587
|
-
*/
|
|
588
|
-
async init() {
|
|
589
|
-
this.ensureDirectories();
|
|
590
|
-
const commits = PostmortemAnalyzer.getFixCommits({
|
|
591
|
-
limit: 200,
|
|
592
|
-
cwd: this.projectRoot
|
|
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 };
|
|
610
|
-
}
|
|
611
|
-
/**
|
|
612
|
-
* 确保目录存在
|
|
613
|
-
*/
|
|
614
|
-
ensureDirectories() {
|
|
615
|
-
const dirs = [
|
|
616
|
-
this.postmortemDir,
|
|
617
|
-
path.join(this.postmortemDir, "categories"),
|
|
618
|
-
path.join(this.postmortemDir, "summaries")
|
|
619
|
-
];
|
|
620
|
-
for (const dir of dirs) {
|
|
621
|
-
if (!nodeFs.existsSync(dir)) {
|
|
622
|
-
nodeFs.mkdirSync(dir, { recursive: true });
|
|
623
|
-
}
|
|
624
|
-
}
|
|
625
|
-
}
|
|
626
|
-
// ==========================================================================
|
|
627
|
-
// Report Management
|
|
628
|
-
// ==========================================================================
|
|
629
|
-
/**
|
|
630
|
-
* 保存 Postmortem 报告
|
|
631
|
-
*/
|
|
632
|
-
saveReport(report) {
|
|
633
|
-
const filename = `${report.id}-${this.slugify(report.title)}.md`;
|
|
634
|
-
const filepath = path.join(this.postmortemDir, filename);
|
|
635
|
-
const content = this.renderReportToMarkdown(report);
|
|
636
|
-
nodeFs.writeFileSync(filepath, content, "utf-8");
|
|
637
|
-
const jsonPath = path.join(this.postmortemDir, `${report.id}.json`);
|
|
638
|
-
nodeFs.writeFileSync(jsonPath, JSON.stringify(report, null, 2), "utf-8");
|
|
639
|
-
return filepath;
|
|
640
|
-
}
|
|
641
|
-
/**
|
|
642
|
-
* 读取 Postmortem 报告
|
|
643
|
-
*/
|
|
644
|
-
getReport(id) {
|
|
645
|
-
const jsonPath = path.join(this.postmortemDir, `${id}.json`);
|
|
646
|
-
if (!nodeFs.existsSync(jsonPath)) {
|
|
647
|
-
return null;
|
|
648
|
-
}
|
|
649
|
-
try {
|
|
650
|
-
const content = nodeFs.readFileSync(jsonPath, "utf-8");
|
|
651
|
-
return JSON.parse(content);
|
|
652
|
-
} catch {
|
|
653
|
-
return null;
|
|
654
|
-
}
|
|
655
|
-
}
|
|
656
|
-
/**
|
|
657
|
-
* 列出所有 Postmortem
|
|
658
|
-
*/
|
|
659
|
-
listReports() {
|
|
660
|
-
const index = this.loadIndex();
|
|
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}"
|
|
672
|
-
};
|
|
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
|
-
}
|
|
740
|
-
/**
|
|
741
|
-
* 生成 slug
|
|
742
|
-
*/
|
|
743
|
-
slugify(text) {
|
|
744
|
-
return text.toLowerCase().replace(/[^\w\s-]/g, "").replace(/\s+/g, "-").substring(0, 50);
|
|
745
|
-
}
|
|
746
|
-
// ==========================================================================
|
|
747
|
-
// Index Management
|
|
748
|
-
// ==========================================================================
|
|
749
|
-
/**
|
|
750
|
-
* 创建空索引
|
|
751
|
-
*/
|
|
752
|
-
createEmptyIndex() {
|
|
753
|
-
return {
|
|
754
|
-
version: "1.0.0",
|
|
755
|
-
lastUpdated: (/* @__PURE__ */ new Date()).toISOString(),
|
|
756
|
-
stats: {
|
|
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: []
|
|
775
|
-
};
|
|
776
|
-
}
|
|
777
|
-
/**
|
|
778
|
-
* 加载索引
|
|
779
|
-
*/
|
|
780
|
-
loadIndex() {
|
|
781
|
-
const indexPath = path.join(this.postmortemDir, INDEX_FILE);
|
|
782
|
-
if (!nodeFs.existsSync(indexPath)) {
|
|
783
|
-
return null;
|
|
784
|
-
}
|
|
785
|
-
try {
|
|
786
|
-
const content = nodeFs.readFileSync(indexPath, "utf-8");
|
|
787
|
-
return JSON.parse(content);
|
|
788
|
-
} catch {
|
|
789
|
-
return null;
|
|
790
|
-
}
|
|
791
|
-
}
|
|
792
|
-
/**
|
|
793
|
-
* 保存索引
|
|
794
|
-
*/
|
|
795
|
-
saveIndex(index) {
|
|
796
|
-
const indexPath = path.join(this.postmortemDir, INDEX_FILE);
|
|
797
|
-
nodeFs.writeFileSync(indexPath, JSON.stringify(index, null, 2), "utf-8");
|
|
798
|
-
}
|
|
799
|
-
/**
|
|
800
|
-
* 更新索引
|
|
801
|
-
*/
|
|
802
|
-
updateIndex() {
|
|
803
|
-
const index = this.createEmptyIndex();
|
|
804
|
-
const files = nodeFs.readdirSync(this.postmortemDir).filter((f) => f.startsWith("PM-") && f.endsWith(".json"));
|
|
805
|
-
for (const file of files) {
|
|
806
|
-
try {
|
|
807
|
-
const content = nodeFs.readFileSync(path.join(this.postmortemDir, file), "utf-8");
|
|
808
|
-
const report = JSON.parse(content);
|
|
809
|
-
index.stats.total++;
|
|
810
|
-
index.stats.bySeverity[report.severity]++;
|
|
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 {
|
|
823
|
-
}
|
|
824
|
-
}
|
|
825
|
-
index.reports.sort((a, b) => {
|
|
826
|
-
const severityOrder = { critical: 0, high: 1, medium: 2, low: 3 };
|
|
827
|
-
const severityDiff = severityOrder[a.severity] - severityOrder[b.severity];
|
|
828
|
-
if (severityDiff !== 0)
|
|
829
|
-
return severityDiff;
|
|
830
|
-
return new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime();
|
|
831
|
-
});
|
|
832
|
-
index.lastUpdated = (/* @__PURE__ */ new Date()).toISOString();
|
|
833
|
-
this.saveIndex(index);
|
|
834
|
-
return index;
|
|
835
|
-
}
|
|
836
|
-
// ==========================================================================
|
|
837
|
-
// CLAUDE.md Integration
|
|
838
|
-
// ==========================================================================
|
|
839
|
-
/**
|
|
840
|
-
* 同步到 CLAUDE.md
|
|
841
|
-
*/
|
|
842
|
-
async syncToClaudeMd() {
|
|
843
|
-
const claudeMdPath = path.join(this.projectRoot, "CLAUDE.md");
|
|
844
|
-
const injection = this.generateClaudeMdInjection();
|
|
845
|
-
let content = "";
|
|
846
|
-
if (nodeFs.existsSync(claudeMdPath)) {
|
|
847
|
-
content = nodeFs.readFileSync(claudeMdPath, "utf-8");
|
|
848
|
-
}
|
|
849
|
-
const startIndex = content.indexOf(CLAUDE_MD_SECTION_START);
|
|
850
|
-
const endIndex = content.indexOf(CLAUDE_MD_SECTION_END);
|
|
851
|
-
if (startIndex !== -1 && endIndex !== -1) {
|
|
852
|
-
content = content.substring(0, startIndex) + content.substring(endIndex + CLAUDE_MD_SECTION_END.length);
|
|
853
|
-
}
|
|
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
|
-
nodeFs.writeFileSync(claudeMdPath, content, "utf-8");
|
|
864
|
-
return {
|
|
865
|
-
synced: injection.sourcePostmortems.length,
|
|
866
|
-
claudeMdPath
|
|
867
|
-
};
|
|
868
|
-
}
|
|
869
|
-
/**
|
|
870
|
-
* 生成 CLAUDE.md 注入内容
|
|
871
|
-
*/
|
|
872
|
-
generateClaudeMdInjection() {
|
|
873
|
-
const index = this.loadIndex();
|
|
874
|
-
const reports = [];
|
|
875
|
-
if (index) {
|
|
876
|
-
const severityOrder = {
|
|
877
|
-
critical: 0,
|
|
878
|
-
high: 1,
|
|
879
|
-
medium: 2,
|
|
880
|
-
low: 3
|
|
881
|
-
};
|
|
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
|
-
}
|
|
895
|
-
const lines = [
|
|
896
|
-
"## \u26A0\uFE0F \u5DF2\u77E5\u95EE\u9898\u9884\u8B66 (Postmortem Intelligence)",
|
|
897
|
-
"",
|
|
898
|
-
"> \u57FA\u4E8E\u5386\u53F2 bug \u5206\u6790\u81EA\u52A8\u751F\u6210\uFF0C\u5E2E\u52A9\u907F\u514D\u91CD\u590D\u72AF\u9519",
|
|
899
|
-
""
|
|
900
|
-
];
|
|
901
|
-
if (reports.length === 0) {
|
|
902
|
-
lines.push("\u6682\u65E0\u9700\u8981\u5173\u6CE8\u7684\u95EE\u9898\u3002");
|
|
903
|
-
} else {
|
|
904
|
-
const critical = reports.filter((r) => r.severity === "critical");
|
|
905
|
-
const high = reports.filter((r) => r.severity === "high");
|
|
906
|
-
const medium = reports.filter((r) => r.severity === "medium");
|
|
907
|
-
if (critical.length > 0) {
|
|
908
|
-
lines.push("### \u{1F534} \u4E25\u91CD");
|
|
909
|
-
for (const r of critical) {
|
|
910
|
-
lines.push(`- **${r.id}**: ${r.title}`);
|
|
911
|
-
lines.push(` - ${r.aiDirectives[0] || r.preventionMeasures[0]}`);
|
|
912
|
-
}
|
|
913
|
-
lines.push("");
|
|
914
|
-
}
|
|
915
|
-
if (high.length > 0) {
|
|
916
|
-
lines.push("### \u{1F7E0} \u9AD8\u4F18\u5148\u7EA7");
|
|
917
|
-
for (const r of high) {
|
|
918
|
-
lines.push(`- **${r.id}**: ${r.title}`);
|
|
919
|
-
lines.push(` - ${r.aiDirectives[0] || r.preventionMeasures[0]}`);
|
|
920
|
-
}
|
|
921
|
-
lines.push("");
|
|
922
|
-
}
|
|
923
|
-
if (medium.length > 0) {
|
|
924
|
-
lines.push("### \u{1F7E1} \u4E2D\u4F18\u5148\u7EA7");
|
|
925
|
-
for (const r of medium) {
|
|
926
|
-
lines.push(`- **${r.id}**: ${r.title}`);
|
|
927
|
-
}
|
|
928
|
-
lines.push("");
|
|
929
|
-
}
|
|
930
|
-
lines.push("### \u{1F4CB} \u5F00\u53D1\u6307\u4EE4");
|
|
931
|
-
const allDirectives = /* @__PURE__ */ new Set();
|
|
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}`);
|
|
939
|
-
}
|
|
940
|
-
lines.push("");
|
|
941
|
-
lines.push(`> \u8BE6\u7EC6\u4FE1\u606F\u8BF7\u67E5\u770B \`${this.config.directory}/\` \u76EE\u5F55`);
|
|
942
|
-
}
|
|
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
|
-
}
|
|
952
|
-
// ==========================================================================
|
|
953
|
-
// Code Checking
|
|
954
|
-
// ==========================================================================
|
|
955
|
-
/**
|
|
956
|
-
* 检查代码是否可能触发已知问题
|
|
957
|
-
*/
|
|
958
|
-
async checkCode(options = {}) {
|
|
959
|
-
const { files, staged } = options;
|
|
960
|
-
let filesToCheck = [];
|
|
961
|
-
if (files && files.length > 0) {
|
|
962
|
-
filesToCheck = files;
|
|
963
|
-
} else if (staged) {
|
|
964
|
-
filesToCheck = this.getStagedFiles();
|
|
965
|
-
} else {
|
|
966
|
-
filesToCheck = this.getAllSourceFiles();
|
|
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 });
|
|
982
|
-
}
|
|
983
|
-
}
|
|
984
|
-
for (const file of filesToCheck) {
|
|
985
|
-
const fullPath = path.isAbsolute(file) ? file : path.join(this.projectRoot, file);
|
|
986
|
-
if (!nodeFs.existsSync(fullPath))
|
|
987
|
-
continue;
|
|
988
|
-
const content = nodeFs.readFileSync(fullPath, "utf-8");
|
|
989
|
-
const lines = content.split("\n");
|
|
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
|
-
}
|
|
94
|
+
async function initI18n(language = "zh-CN") {
|
|
95
|
+
if (i18n.isInitialized) {
|
|
96
|
+
if (i18n.language !== language) {
|
|
97
|
+
await i18n.changeLanguage(language);
|
|
98
|
+
}
|
|
99
|
+
return;
|
|
100
|
+
}
|
|
101
|
+
await i18n.use(Backend).init({
|
|
102
|
+
lng: language,
|
|
103
|
+
fallbackLng: "en",
|
|
104
|
+
// Load all translations as a single flat structure
|
|
105
|
+
ns: NAMESPACES,
|
|
106
|
+
defaultNS: "common",
|
|
107
|
+
preload: [language],
|
|
108
|
+
// Preload the selected language
|
|
109
|
+
// Suppress i18next locize promotional banner in CLI output
|
|
110
|
+
partialBundledLanguages: true,
|
|
111
|
+
// Backend configuration for loading JSON files
|
|
112
|
+
backend: {
|
|
113
|
+
loadPath: (() => {
|
|
114
|
+
const currentDir = dirname(fileURLToPath(import.meta.url));
|
|
115
|
+
const packageRoot = (() => {
|
|
116
|
+
let dir = currentDir;
|
|
117
|
+
while (dir !== dirname(dir)) {
|
|
118
|
+
if (existsSync(join(dir, "package.json"))) {
|
|
119
|
+
return dir;
|
|
1011
120
|
}
|
|
1012
|
-
|
|
121
|
+
dir = dirname(dir);
|
|
1013
122
|
}
|
|
1014
|
-
|
|
1015
|
-
|
|
1016
|
-
|
|
1017
|
-
|
|
1018
|
-
|
|
1019
|
-
|
|
1020
|
-
|
|
1021
|
-
|
|
1022
|
-
|
|
1023
|
-
|
|
1024
|
-
|
|
1025
|
-
|
|
1026
|
-
|
|
1027
|
-
|
|
1028
|
-
|
|
1029
|
-
|
|
1030
|
-
|
|
1031
|
-
|
|
1032
|
-
* 获取暂存的文件
|
|
1033
|
-
*/
|
|
1034
|
-
getStagedFiles() {
|
|
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 [];
|
|
1043
|
-
}
|
|
1044
|
-
}
|
|
1045
|
-
/**
|
|
1046
|
-
* 获取所有源文件
|
|
1047
|
-
*/
|
|
1048
|
-
getAllSourceFiles() {
|
|
1049
|
-
const files = [];
|
|
1050
|
-
const walk = (dir) => {
|
|
1051
|
-
const entries = nodeFs.readdirSync(dir, { withFileTypes: true });
|
|
1052
|
-
for (const entry of entries) {
|
|
1053
|
-
const fullPath = path.join(dir, entry.name);
|
|
1054
|
-
const relativePath = path.relative(this.projectRoot, fullPath);
|
|
1055
|
-
if (this.config.detection.excludePatterns.some((p) => this.matchGlob(relativePath, p))) {
|
|
1056
|
-
continue;
|
|
1057
|
-
}
|
|
1058
|
-
if (entry.isDirectory()) {
|
|
1059
|
-
walk(fullPath);
|
|
1060
|
-
} else if (entry.isFile()) {
|
|
1061
|
-
if (this.config.detection.includePatterns.some((p) => this.matchGlob(relativePath, p))) {
|
|
1062
|
-
files.push(relativePath);
|
|
123
|
+
return currentDir;
|
|
124
|
+
})();
|
|
125
|
+
const possibleBasePaths = [
|
|
126
|
+
join(currentDir, "locales"),
|
|
127
|
+
// Development: src/i18n/locales
|
|
128
|
+
join(packageRoot, "dist/i18n/locales"),
|
|
129
|
+
// NPM package: /node_modules/ccjk/dist/i18n/locales
|
|
130
|
+
join(process__default.cwd(), "dist/i18n/locales"),
|
|
131
|
+
// Production build: ./dist/i18n/locales
|
|
132
|
+
join(currentDir, "../../../dist/i18n/locales"),
|
|
133
|
+
// Fallback for deep chunk paths
|
|
134
|
+
join(currentDir, "../../i18n/locales")
|
|
135
|
+
// Alternative chunk structure
|
|
136
|
+
];
|
|
137
|
+
for (const basePath of possibleBasePaths) {
|
|
138
|
+
const testFile = join(basePath, "zh-CN/common.json");
|
|
139
|
+
if (existsSync(testFile)) {
|
|
140
|
+
return join(basePath, "{{lng}}/{{ns}}.json");
|
|
1063
141
|
}
|
|
1064
142
|
}
|
|
1065
|
-
|
|
1066
|
-
|
|
1067
|
-
|
|
1068
|
-
|
|
1069
|
-
|
|
1070
|
-
|
|
1071
|
-
|
|
1072
|
-
|
|
1073
|
-
|
|
1074
|
-
|
|
1075
|
-
|
|
1076
|
-
|
|
1077
|
-
|
|
1078
|
-
|
|
1079
|
-
|
|
1080
|
-
|
|
1081
|
-
|
|
1082
|
-
|
|
1083
|
-
|
|
1084
|
-
|
|
1085
|
-
|
|
1086
|
-
|
|
1087
|
-
|
|
1088
|
-
|
|
1089
|
-
|
|
1090
|
-
|
|
1091
|
-
|
|
1092
|
-
|
|
1093
|
-
|
|
1094
|
-
|
|
1095
|
-
|
|
1096
|
-
|
|
1097
|
-
const commits = PostmortemAnalyzer.getFixCommits({
|
|
1098
|
-
since,
|
|
1099
|
-
until,
|
|
1100
|
-
cwd: this.projectRoot
|
|
1101
|
-
});
|
|
1102
|
-
const analyses = commits.map(
|
|
1103
|
-
(c) => PostmortemAnalyzer.analyzeFixCommit(c, this.projectRoot)
|
|
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);
|
|
1112
|
-
}
|
|
1113
|
-
this.updateIndex();
|
|
1114
|
-
const summary = {
|
|
1115
|
-
version,
|
|
1116
|
-
releaseDate: (/* @__PURE__ */ new Date()).toISOString(),
|
|
1117
|
-
fixCommitCount: commits.length,
|
|
1118
|
-
newPostmortems: newIds,
|
|
1119
|
-
updatedPostmortems: [],
|
|
1120
|
-
summary: this.generateReleaseSummaryText(commits, newReports),
|
|
1121
|
-
keyLessons: this.extractKeyLessons(newReports)
|
|
1122
|
-
};
|
|
1123
|
-
const summaryPath = path.join(this.postmortemDir, "summaries", `${version}.json`);
|
|
1124
|
-
nodeFs.writeFileSync(summaryPath, JSON.stringify(summary, null, 2), "utf-8");
|
|
1125
|
-
if (this.config.autoSyncToClaudeMd) {
|
|
1126
|
-
await this.syncToClaudeMd();
|
|
1127
|
-
}
|
|
1128
|
-
return summary;
|
|
1129
|
-
}
|
|
1130
|
-
/**
|
|
1131
|
-
* 生成发布摘要文本
|
|
1132
|
-
*/
|
|
1133
|
-
generateReleaseSummaryText(commits, reports) {
|
|
1134
|
-
const lines = [
|
|
1135
|
-
`\u672C\u6B21\u53D1\u5E03\u5305\u542B ${commits.length} \u4E2A bug \u4FEE\u590D\uFF0C\u751F\u6210\u4E86 ${reports.length} \u4E2A\u65B0\u7684 Postmortem \u62A5\u544A\u3002`,
|
|
1136
|
-
""
|
|
1137
|
-
];
|
|
1138
|
-
if (reports.length > 0) {
|
|
1139
|
-
lines.push("\u4E3B\u8981\u95EE\u9898\u7C7B\u578B:");
|
|
1140
|
-
const categories = /* @__PURE__ */ new Map();
|
|
1141
|
-
for (const r of reports) {
|
|
1142
|
-
categories.set(r.category, (categories.get(r.category) || 0) + 1);
|
|
1143
|
-
}
|
|
1144
|
-
for (const [cat, count] of categories) {
|
|
1145
|
-
lines.push(`- ${cat}: ${count} \u4E2A`);
|
|
1146
|
-
}
|
|
1147
|
-
}
|
|
1148
|
-
return lines.join("\n");
|
|
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
|
-
}
|
|
1159
|
-
}
|
|
1160
|
-
return Array.from(lessons).slice(0, 10);
|
|
1161
|
-
}
|
|
1162
|
-
}
|
|
1163
|
-
let managerInstance = null;
|
|
1164
|
-
function getPostmortemManager(projectRoot, config) {
|
|
1165
|
-
if (!managerInstance || projectRoot) {
|
|
1166
|
-
managerInstance = new PostmortemManager(projectRoot, config);
|
|
1167
|
-
}
|
|
1168
|
-
return managerInstance;
|
|
143
|
+
return join(process__default.cwd(), "dist/i18n/locales/{{lng}}/{{ns}}.json");
|
|
144
|
+
})()
|
|
145
|
+
},
|
|
146
|
+
// Interpolation settings
|
|
147
|
+
interpolation: {
|
|
148
|
+
escapeValue: false
|
|
149
|
+
// Not needed for server-side usage
|
|
150
|
+
},
|
|
151
|
+
// Enable key separator for nested keys, enable namespace separator
|
|
152
|
+
keySeparator: ".",
|
|
153
|
+
nsSeparator: ":",
|
|
154
|
+
// Debugging (disable for clean output)
|
|
155
|
+
debug: false
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
function format(template, values) {
|
|
159
|
+
if (!values)
|
|
160
|
+
return template;
|
|
161
|
+
return Object.keys(values).reduce((result, key) => {
|
|
162
|
+
return result.replace(new RegExp(`{${key}}`, "g"), values[key]);
|
|
163
|
+
}, template);
|
|
164
|
+
}
|
|
165
|
+
async function changeLanguage(lng) {
|
|
166
|
+
await i18n.changeLanguage(lng);
|
|
167
|
+
}
|
|
168
|
+
function getTranslation(_lang) {
|
|
169
|
+
return (key, options) => {
|
|
170
|
+
if (key.includes(":")) {
|
|
171
|
+
return i18n.t(key, options);
|
|
172
|
+
}
|
|
173
|
+
return i18n.t(`common:${key}`, options);
|
|
174
|
+
};
|
|
1169
175
|
}
|
|
1170
176
|
|
|
1171
|
-
export {
|
|
177
|
+
export { changeLanguage, ensureI18nInitialized, format, getTranslation, i18n, initI18n };
|