frontend-guardian-core 2.6.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.
Files changed (152) hide show
  1. package/LICENSE +21 -0
  2. package/bin/fg-core.js +1238 -0
  3. package/bin/watch-mode.js +123 -0
  4. package/dist/engine/cache.d.ts +68 -0
  5. package/dist/engine/cache.d.ts.map +1 -0
  6. package/dist/engine/cache.js +164 -0
  7. package/dist/engine/cache.js.map +1 -0
  8. package/dist/engine/rule-engine.d.ts +135 -0
  9. package/dist/engine/rule-engine.d.ts.map +1 -0
  10. package/dist/engine/rule-engine.js +716 -0
  11. package/dist/engine/rule-engine.js.map +1 -0
  12. package/dist/formatters/github-annotation.d.ts +36 -0
  13. package/dist/formatters/github-annotation.d.ts.map +1 -0
  14. package/dist/formatters/github-annotation.js +122 -0
  15. package/dist/formatters/github-annotation.js.map +1 -0
  16. package/dist/formatters/pr-comment.d.ts +43 -0
  17. package/dist/formatters/pr-comment.d.ts.map +1 -0
  18. package/dist/formatters/pr-comment.js +171 -0
  19. package/dist/formatters/pr-comment.js.map +1 -0
  20. package/dist/formatters/sarif.d.ts +104 -0
  21. package/dist/formatters/sarif.d.ts.map +1 -0
  22. package/dist/formatters/sarif.js +130 -0
  23. package/dist/formatters/sarif.js.map +1 -0
  24. package/dist/index.d.ts +46 -0
  25. package/dist/index.d.ts.map +1 -0
  26. package/dist/index.js +108 -0
  27. package/dist/index.js.map +1 -0
  28. package/dist/integrations/base.d.ts +44 -0
  29. package/dist/integrations/base.d.ts.map +1 -0
  30. package/dist/integrations/base.js +104 -0
  31. package/dist/integrations/base.js.map +1 -0
  32. package/dist/integrations/eslint.d.ts +8 -0
  33. package/dist/integrations/eslint.d.ts.map +1 -0
  34. package/dist/integrations/eslint.js +67 -0
  35. package/dist/integrations/eslint.js.map +1 -0
  36. package/dist/integrations/formatter.d.ts +35 -0
  37. package/dist/integrations/formatter.d.ts.map +1 -0
  38. package/dist/integrations/formatter.js +182 -0
  39. package/dist/integrations/formatter.js.map +1 -0
  40. package/dist/integrations/index.d.ts +17 -0
  41. package/dist/integrations/index.d.ts.map +1 -0
  42. package/dist/integrations/index.js +25 -0
  43. package/dist/integrations/index.js.map +1 -0
  44. package/dist/integrations/stylelint.d.ts +8 -0
  45. package/dist/integrations/stylelint.d.ts.map +1 -0
  46. package/dist/integrations/stylelint.js +59 -0
  47. package/dist/integrations/stylelint.js.map +1 -0
  48. package/dist/integrations/typescript.d.ts +8 -0
  49. package/dist/integrations/typescript.d.ts.map +1 -0
  50. package/dist/integrations/typescript.js +92 -0
  51. package/dist/integrations/typescript.js.map +1 -0
  52. package/dist/rules/registry.d.ts +83 -0
  53. package/dist/rules/registry.d.ts.map +1 -0
  54. package/dist/rules/registry.js +205 -0
  55. package/dist/rules/registry.js.map +1 -0
  56. package/dist/scanners/a11y-scanner.d.ts +14 -0
  57. package/dist/scanners/a11y-scanner.d.ts.map +1 -0
  58. package/dist/scanners/a11y-scanner.js +781 -0
  59. package/dist/scanners/a11y-scanner.js.map +1 -0
  60. package/dist/scanners/component-scanner.d.ts +12 -0
  61. package/dist/scanners/component-scanner.d.ts.map +1 -0
  62. package/dist/scanners/component-scanner.js +304 -0
  63. package/dist/scanners/component-scanner.js.map +1 -0
  64. package/dist/scanners/cross-file-scanner.d.ts +18 -0
  65. package/dist/scanners/cross-file-scanner.d.ts.map +1 -0
  66. package/dist/scanners/cross-file-scanner.js +684 -0
  67. package/dist/scanners/cross-file-scanner.js.map +1 -0
  68. package/dist/scanners/hooks-scanner.d.ts +15 -0
  69. package/dist/scanners/hooks-scanner.d.ts.map +1 -0
  70. package/dist/scanners/hooks-scanner.js +670 -0
  71. package/dist/scanners/hooks-scanner.js.map +1 -0
  72. package/dist/scanners/i18n-scanner.d.ts +13 -0
  73. package/dist/scanners/i18n-scanner.d.ts.map +1 -0
  74. package/dist/scanners/i18n-scanner.js +535 -0
  75. package/dist/scanners/i18n-scanner.js.map +1 -0
  76. package/dist/scanners/naming-scanner.d.ts +19 -0
  77. package/dist/scanners/naming-scanner.d.ts.map +1 -0
  78. package/dist/scanners/naming-scanner.js +746 -0
  79. package/dist/scanners/naming-scanner.js.map +1 -0
  80. package/dist/scanners/performance-scanner.d.ts +7 -0
  81. package/dist/scanners/performance-scanner.d.ts.map +1 -0
  82. package/dist/scanners/performance-scanner.js +402 -0
  83. package/dist/scanners/performance-scanner.js.map +1 -0
  84. package/dist/scanners/platform-scanner.d.ts +15 -0
  85. package/dist/scanners/platform-scanner.d.ts.map +1 -0
  86. package/dist/scanners/platform-scanner.js +320 -0
  87. package/dist/scanners/platform-scanner.js.map +1 -0
  88. package/dist/scanners/security-scanner.d.ts +7 -0
  89. package/dist/scanners/security-scanner.d.ts.map +1 -0
  90. package/dist/scanners/security-scanner.js +349 -0
  91. package/dist/scanners/security-scanner.js.map +1 -0
  92. package/dist/scanners/svelte-scanner.d.ts +14 -0
  93. package/dist/scanners/svelte-scanner.d.ts.map +1 -0
  94. package/dist/scanners/svelte-scanner.js +228 -0
  95. package/dist/scanners/svelte-scanner.js.map +1 -0
  96. package/dist/types.d.ts +343 -0
  97. package/dist/types.d.ts.map +1 -0
  98. package/dist/types.js +6 -0
  99. package/dist/types.js.map +1 -0
  100. package/dist/utils/ast-parser.d.ts +21 -0
  101. package/dist/utils/ast-parser.d.ts.map +1 -0
  102. package/dist/utils/ast-parser.js +119 -0
  103. package/dist/utils/ast-parser.js.map +1 -0
  104. package/dist/utils/baseline.d.ts +89 -0
  105. package/dist/utils/baseline.d.ts.map +1 -0
  106. package/dist/utils/baseline.js +156 -0
  107. package/dist/utils/baseline.js.map +1 -0
  108. package/dist/utils/ci-generator.d.ts +34 -0
  109. package/dist/utils/ci-generator.d.ts.map +1 -0
  110. package/dist/utils/ci-generator.js +194 -0
  111. package/dist/utils/ci-generator.js.map +1 -0
  112. package/dist/utils/common.d.ts +8 -0
  113. package/dist/utils/common.d.ts.map +1 -0
  114. package/dist/utils/common.js +38 -0
  115. package/dist/utils/common.js.map +1 -0
  116. package/dist/utils/concurrent.d.ts +16 -0
  117. package/dist/utils/concurrent.d.ts.map +1 -0
  118. package/dist/utils/concurrent.js +49 -0
  119. package/dist/utils/concurrent.js.map +1 -0
  120. package/dist/utils/config-loader.d.ts +8 -0
  121. package/dist/utils/config-loader.d.ts.map +1 -0
  122. package/dist/utils/config-loader.js +154 -0
  123. package/dist/utils/config-loader.js.map +1 -0
  124. package/dist/utils/fix-bot.d.ts +36 -0
  125. package/dist/utils/fix-bot.d.ts.map +1 -0
  126. package/dist/utils/fix-bot.js +274 -0
  127. package/dist/utils/fix-bot.js.map +1 -0
  128. package/dist/utils/git-hooks.d.ts +55 -0
  129. package/dist/utils/git-hooks.d.ts.map +1 -0
  130. package/dist/utils/git-hooks.js +318 -0
  131. package/dist/utils/git-hooks.js.map +1 -0
  132. package/dist/utils/history-report.d.ts +72 -0
  133. package/dist/utils/history-report.d.ts.map +1 -0
  134. package/dist/utils/history-report.js +144 -0
  135. package/dist/utils/history-report.js.map +1 -0
  136. package/dist/utils/init-config.d.ts +23 -0
  137. package/dist/utils/init-config.d.ts.map +1 -0
  138. package/dist/utils/init-config.js +146 -0
  139. package/dist/utils/init-config.js.map +1 -0
  140. package/dist/utils/pr-publisher.d.ts +64 -0
  141. package/dist/utils/pr-publisher.d.ts.map +1 -0
  142. package/dist/utils/pr-publisher.js +265 -0
  143. package/dist/utils/pr-publisher.js.map +1 -0
  144. package/dist/utils/project-detector.d.ts +20 -0
  145. package/dist/utils/project-detector.d.ts.map +1 -0
  146. package/dist/utils/project-detector.js +342 -0
  147. package/dist/utils/project-detector.js.map +1 -0
  148. package/dist/utils/report-uploader.d.ts +35 -0
  149. package/dist/utils/report-uploader.d.ts.map +1 -0
  150. package/dist/utils/report-uploader.js +106 -0
  151. package/dist/utils/report-uploader.js.map +1 -0
  152. package/package.json +78 -0
@@ -0,0 +1,318 @@
1
+ "use strict";
2
+ /**
3
+ * Git Hook 自动安装工具
4
+ * 一键配置 pre-commit / pre-push hook,在提交前自动运行增量扫描
5
+ */
6
+ var __importDefault = (this && this.__importDefault) || function (mod) {
7
+ return (mod && mod.__esModule) ? mod : { "default": mod };
8
+ };
9
+ Object.defineProperty(exports, "__esModule", { value: true });
10
+ exports.detectHusky = detectHusky;
11
+ exports.installGitHooks = installGitHooks;
12
+ exports.uninstallGitHooks = uninstallGitHooks;
13
+ exports.hasGitHook = hasGitHook;
14
+ const node_fs_1 = require("node:fs");
15
+ const node_path_1 = require("node:path");
16
+ const node_child_process_1 = require("node:child_process");
17
+ const picocolors_1 = __importDefault(require("picocolors"));
18
+ const HOOK_TEMPLATE_PRE_COMMIT = `#!/bin/sh
19
+ # Auto-generated by frontend-guardian
20
+ # 提交前自动运行增量扫描(仅 staged 文件)
21
+
22
+ echo "🔍 Frontend Guardian: 正在检查提交文件..."
23
+
24
+ npx fg-core . --scan --staged{{ARGS}}
25
+
26
+ EXIT_CODE=$?
27
+ if [ $EXIT_CODE -ne 0 ]; then
28
+ echo "❌ 检查未通过,请修复问题后再提交"
29
+ echo " 使用 --no-verify 跳过检查(不推荐)"
30
+ exit 1
31
+ fi
32
+
33
+ echo "✅ 检查通过"
34
+ `;
35
+ const HOOK_TEMPLATE_PRE_PUSH = `#!/bin/sh
36
+ # Auto-generated by frontend-guardian
37
+ # 推送前自动运行全量扫描
38
+
39
+ echo "🔍 Frontend Guardian: 正在运行全量扫描..."
40
+
41
+ npx fg-core . --scan{{ARGS}}
42
+
43
+ EXIT_CODE=$?
44
+ if [ $EXIT_CODE -ne 0 ]; then
45
+ echo "❌ 检查未通过,请修复问题后再推送"
46
+ echo " 使用 --no-verify 跳过检查(不推荐)"
47
+ exit 1
48
+ fi
49
+
50
+ echo "✅ 检查通过"
51
+ `;
52
+ const HOOK_TEMPLATE_COMMIT_MSG = `#!/bin/sh
53
+ # Auto-generated by frontend-guardian
54
+ # 检查 commit message 是否符合 Conventional Commits 规范
55
+
56
+ COMMIT_MSG_FILE=$1
57
+ COMMIT_MSG=$(cat "$COMMIT_MSG_FILE")
58
+
59
+ # 跳过 merge commit、revert、amend
60
+ if echo "$COMMIT_MSG" | grep -qE "^Merge |^Revert |^fixup! |^squash! "; then
61
+ exit 0
62
+ fi
63
+
64
+ # 跳过空消息检查(如果配置允许)
65
+ {{ALLOW_EMPTY}}
66
+
67
+ # Conventional Commits 正则
68
+ PATTERN="{{PATTERN}}"
69
+
70
+ if ! echo "$COMMIT_MSG" | grep -qE "$PATTERN"; then
71
+ echo "❌ Commit message 格式不符合规范"
72
+ echo ""
73
+ echo " 期望格式: <type>[(scope)]: <subject>"
74
+ echo ""
75
+ echo " 支持的 type:"
76
+ echo " feat 新功能"
77
+ echo " fix 修复"
78
+ echo " docs 文档"
79
+ echo " style 代码格式"
80
+ echo " refactor 重构"
81
+ echo " perf 性能优化"
82
+ echo " test 测试"
83
+ echo " build 构建"
84
+ echo " ci CI/CD"
85
+ echo " chore 杂项"
86
+ echo " revert 回滚"
87
+ echo ""
88
+ echo " 示例:"
89
+ echo " feat(auth): 添加登录页面"
90
+ echo " fix: 修复空指针异常"
91
+ echo ""
92
+ echo " 当前消息:"
93
+ echo " $COMMIT_MSG"
94
+ exit 1
95
+ fi
96
+
97
+ echo "✅ Commit message 格式检查通过"
98
+ `;
99
+ /**
100
+ * 检测项目是否使用 husky
101
+ * @param projectDir 项目根目录
102
+ * @returns husky 配置信息
103
+ */
104
+ function detectHusky(projectDir) {
105
+ const huskyDirV8 = (0, node_path_1.resolve)(projectDir, ".husky");
106
+ const huskyDirLegacy = (0, node_path_1.resolve)(projectDir, ".husky/_/husky.sh");
107
+ if ((0, node_fs_1.existsSync)(huskyDirV8)) {
108
+ return { useHusky: true, huskyDir: huskyDirV8, version: "v8+" };
109
+ }
110
+ if ((0, node_fs_1.existsSync)(huskyDirLegacy)) {
111
+ return { useHusky: true, huskyDir: huskyDirV8, version: "legacy" };
112
+ }
113
+ return { useHusky: false, huskyDir: null, version: null };
114
+ }
115
+ /**
116
+ * 安装 Git Hook
117
+ * @param projectDir 项目根目录
118
+ * @param config Hook 配置
119
+ * @returns 安装结果
120
+ */
121
+ function installGitHooks(projectDir, config) {
122
+ const installed = [];
123
+ const skipped = [];
124
+ // 查找 git 根目录
125
+ let gitDir;
126
+ try {
127
+ const gitRoot = (0, node_child_process_1.execSync)("git rev-parse --git-dir", {
128
+ cwd: projectDir,
129
+ encoding: "utf-8",
130
+ stdio: ["pipe", "pipe", "ignore"],
131
+ }).trim();
132
+ gitDir = (0, node_path_1.resolve)(projectDir, gitRoot);
133
+ }
134
+ catch {
135
+ console.log(picocolors_1.default.yellow("⚠️ 未找到 Git 仓库,跳过 Hook 安装"));
136
+ return { installed, skipped };
137
+ }
138
+ // 检测 husky
139
+ const huskyInfo = detectHusky(projectDir);
140
+ const useHusky = huskyInfo.useHusky;
141
+ const hooksDir = useHusky && huskyInfo.huskyDir
142
+ ? huskyInfo.huskyDir
143
+ : (0, node_path_1.resolve)(gitDir, "hooks");
144
+ if (!(0, node_fs_1.existsSync)(hooksDir)) {
145
+ (0, node_fs_1.mkdirSync)(hooksDir, { recursive: true });
146
+ }
147
+ const types = (() => {
148
+ switch (config.type) {
149
+ case "both": return ["pre-commit", "pre-push"];
150
+ case "all": return ["pre-commit", "pre-push", "commit-msg"];
151
+ default: return [config.type];
152
+ }
153
+ })();
154
+ // 构建参数
155
+ const args = [];
156
+ if (config.autoFix)
157
+ args.push(" --fix");
158
+ if (config.cache === false)
159
+ args.push(" --no-cache");
160
+ if (config.minSeverity)
161
+ args.push(` --severity ${config.minSeverity}`);
162
+ const argsStr = args.join("");
163
+ for (const hookType of types) {
164
+ const hookPath = (0, node_path_1.resolve)(hooksDir, hookType);
165
+ let template;
166
+ if (hookType === "pre-push") {
167
+ template = HOOK_TEMPLATE_PRE_PUSH;
168
+ }
169
+ else if (hookType === "commit-msg") {
170
+ template = HOOK_TEMPLATE_COMMIT_MSG;
171
+ }
172
+ else {
173
+ template = HOOK_TEMPLATE_PRE_COMMIT;
174
+ }
175
+ // 检查是否已有 hook
176
+ if ((0, node_fs_1.existsSync)(hookPath)) {
177
+ const existing = (0, node_fs_1.readFileSync)(hookPath, "utf-8");
178
+ if (existing.includes("frontend-guardian")) {
179
+ // 已安装过,更新内容
180
+ const newContent = buildHookContent(template, argsStr, hookType, config, useHusky);
181
+ (0, node_fs_1.writeFileSync)(hookPath, newContent, "utf-8");
182
+ if (!useHusky)
183
+ (0, node_fs_1.chmodSync)(hookPath, 0o755);
184
+ installed.push(`${hookType}${useHusky ? " (husky)" : ""} (更新)`);
185
+ continue;
186
+ }
187
+ // 已有其他 hook,询问是否合并(此处静默跳过,实际使用时应提示用户)
188
+ skipped.push(`${hookType}${useHusky ? " (husky)" : ""} (已有其他 hook)`);
189
+ continue;
190
+ }
191
+ // 新建 hook
192
+ const content = buildHookContent(template, argsStr, hookType, config, useHusky);
193
+ (0, node_fs_1.writeFileSync)(hookPath, content, "utf-8");
194
+ if (!useHusky)
195
+ (0, node_fs_1.chmodSync)(hookPath, 0o755);
196
+ installed.push(`${hookType}${useHusky ? " (husky)" : ""}`);
197
+ }
198
+ return { installed, skipped };
199
+ }
200
+ /**
201
+ * 构建 hook 内容,根据类型应用不同替换
202
+ */
203
+ function buildHookContent(template, argsStr, hookType, config, useHusky) {
204
+ let content = template;
205
+ if (hookType === "commit-msg") {
206
+ // Conventional Commits 默认正则
207
+ const pattern = config.commitMsgPattern ||
208
+ "^(feat|fix|docs|style|refactor|perf|test|build|ci|chore|revert)(\\(.+\\))?!?: .+";
209
+ const allowEmpty = config.allowEmpty
210
+ ? "# 允许空消息\nif [ -z \"$COMMIT_MSG\" ] || [ \"$COMMIT_MSG\" = \"\" ]; then\n exit 0\nfi\n"
211
+ : "# 空消息检查(默认拒绝)\nif [ -z \"$COMMIT_MSG\" ] || [ \"$COMMIT_MSG\" = \"\" ]; then\n echo \"❌ Commit message 不能为空\"\n exit 1\nfi\n";
212
+ content = content.replace("{{PATTERN}}", pattern);
213
+ content = content.replace("{{ALLOW_EMPTY}}", allowEmpty);
214
+ }
215
+ else {
216
+ content = content.replace("{{ARGS}}", argsStr);
217
+ }
218
+ if (useHusky) {
219
+ content = toHuskyScript(content, argsStr);
220
+ }
221
+ return content;
222
+ }
223
+ /**
224
+ * 将 shell hook 模板转换为 husky v8+ 脚本格式
225
+ */
226
+ function toHuskyScript(template, argsStr) {
227
+ // 移除 shebang,husky v8+ 不需要
228
+ const lines = template
229
+ .replace("#!/bin/sh\n", "")
230
+ .split("\n")
231
+ .filter((line) => !line.startsWith("# ") || line.includes("frontend-guardian"));
232
+ // 替换命令中的 {{ARGS}}
233
+ return lines
234
+ .join("\n")
235
+ .replace("{{ARGS}}", argsStr)
236
+ .trim();
237
+ }
238
+ /**
239
+ * 卸载 Git Hook
240
+ * @param projectDir 项目根目录
241
+ * @param type 要卸载的 hook 类型
242
+ * @returns 卸载结果
243
+ */
244
+ function uninstallGitHooks(projectDir, type) {
245
+ const removed = [];
246
+ const notFound = [];
247
+ let gitDir;
248
+ try {
249
+ const gitRoot = (0, node_child_process_1.execSync)("git rev-parse --git-dir", {
250
+ cwd: projectDir,
251
+ encoding: "utf-8",
252
+ stdio: ["pipe", "pipe", "ignore"],
253
+ }).trim();
254
+ gitDir = (0, node_path_1.resolve)(projectDir, gitRoot);
255
+ }
256
+ catch {
257
+ return { removed, notFound };
258
+ }
259
+ const huskyInfo = detectHusky(projectDir);
260
+ const hooksDir = huskyInfo.useHusky && huskyInfo.huskyDir
261
+ ? huskyInfo.huskyDir
262
+ : (0, node_path_1.resolve)(gitDir, "hooks");
263
+ const types = (() => {
264
+ switch (type) {
265
+ case "both": return ["pre-commit", "pre-push"];
266
+ case "all": return ["pre-commit", "pre-push", "commit-msg"];
267
+ default: return [type];
268
+ }
269
+ })();
270
+ for (const hookType of types) {
271
+ const hookPath = (0, node_path_1.resolve)(hooksDir, hookType);
272
+ if ((0, node_fs_1.existsSync)(hookPath)) {
273
+ const content = (0, node_fs_1.readFileSync)(hookPath, "utf-8");
274
+ if (content.includes("frontend-guardian")) {
275
+ const fs = require("node:fs");
276
+ fs.unlinkSync(hookPath);
277
+ removed.push(`${hookType}${huskyInfo.useHusky ? " (husky)" : ""}`);
278
+ }
279
+ else {
280
+ notFound.push(`${hookType}${huskyInfo.useHusky ? " (husky)" : ""} (非 fg 生成)`);
281
+ }
282
+ }
283
+ else {
284
+ notFound.push(hookType);
285
+ }
286
+ }
287
+ return { removed, notFound };
288
+ }
289
+ /**
290
+ * 检查是否已安装 fg hook
291
+ */
292
+ function hasGitHook(projectDir, type) {
293
+ try {
294
+ const huskyInfo = detectHusky(projectDir);
295
+ if (huskyInfo.useHusky && huskyInfo.huskyDir) {
296
+ const hookPath = (0, node_path_1.resolve)(huskyInfo.huskyDir, type);
297
+ if ((0, node_fs_1.existsSync)(hookPath)) {
298
+ const content = (0, node_fs_1.readFileSync)(hookPath, "utf-8");
299
+ return content.includes("frontend-guardian");
300
+ }
301
+ return false;
302
+ }
303
+ const gitRoot = (0, node_child_process_1.execSync)("git rev-parse --git-dir", {
304
+ cwd: projectDir,
305
+ encoding: "utf-8",
306
+ stdio: "pipe",
307
+ }).trim();
308
+ const hookPath = (0, node_path_1.resolve)(projectDir, gitRoot, "hooks", type);
309
+ if (!(0, node_fs_1.existsSync)(hookPath))
310
+ return false;
311
+ const content = (0, node_fs_1.readFileSync)(hookPath, "utf-8");
312
+ return content.includes("frontend-guardian");
313
+ }
314
+ catch {
315
+ return false;
316
+ }
317
+ }
318
+ //# sourceMappingURL=git-hooks.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"git-hooks.js","sourceRoot":"","sources":["../../src/utils/git-hooks.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,72 @@
1
+ /**
2
+ * History Report — 扫描历史记录与趋势分析
3
+ *
4
+ * 功能:
5
+ * 1. 每次扫描后自动记录结果
6
+ * 2. 对比历史数据,高亮新问题 / 已修复问题
7
+ * 3. 生成趋势图表数据(JSON)
8
+ * 4. 团队治理追踪
9
+ */
10
+ import type { Issue, ScanResult } from "../types.js";
11
+ export interface HistoryEntry {
12
+ /** 扫描时间戳 */
13
+ timestamp: number;
14
+ /** 扫描模块 */
15
+ module: string;
16
+ /** 各严重级别问题数 */
17
+ counts: {
18
+ critical: number;
19
+ warning: number;
20
+ suggestion: number;
21
+ };
22
+ /** 问题签名列表(用于对比) */
23
+ signatures: string[];
24
+ /** 扫描文件数 */
25
+ filesScanned: number;
26
+ /** 扫描耗时 ms */
27
+ duration: number;
28
+ /** 提交 hash(如果有) */
29
+ commit?: string;
30
+ /** 分支 */
31
+ branch?: string;
32
+ }
33
+ export interface TrendAnalysis {
34
+ /** 总扫描次数 */
35
+ totalScans: number;
36
+ /** 最新扫描时间 */
37
+ lastScan: number;
38
+ /** 趋势:各严重级别问题数变化 */
39
+ trend: Array<{
40
+ timestamp: number;
41
+ critical: number;
42
+ warning: number;
43
+ suggestion: number;
44
+ }>;
45
+ /** 新问题(本次 vs 上次) */
46
+ newIssues: string[];
47
+ /** 已修复问题(上次存在本次不存在) */
48
+ fixedIssues: string[];
49
+ /** 持续存在的问题 */
50
+ persistentIssues: string[];
51
+ }
52
+ export declare class HistoryReport {
53
+ private historyDir;
54
+ private historyFile;
55
+ private entries;
56
+ constructor(projectDir: string);
57
+ /**
58
+ * 记录一次扫描结果
59
+ */
60
+ record(result: ScanResult, allIssues: Issue[]): void;
61
+ /**
62
+ * 分析趋势:对比本次与上一次同模块扫描
63
+ */
64
+ analyze(module: string, currentSignatures: string[]): TrendAnalysis;
65
+ /** 获取所有历史记录 */
66
+ getEntries(): HistoryEntry[];
67
+ /** 清空历史 */
68
+ clear(): void;
69
+ private loadEntries;
70
+ private save;
71
+ }
72
+ //# sourceMappingURL=history-report.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"history-report.d.ts","sourceRoot":"","sources":["../../src/utils/history-report.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAIH,OAAO,KAAK,EAAE,KAAK,EAAE,UAAU,EAAE,MAAM,YAAY,CAAC;AAEpD,MAAM,WAAW,YAAY;IACzB,YAAY;IACZ,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW;IACX,MAAM,EAAE,MAAM,CAAC;IACf,eAAe;IACf,MAAM,EAAE;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAC;QAAC,UAAU,EAAE,MAAM,CAAA;KAAE,CAAC;IAClE,mBAAmB;IACnB,UAAU,EAAE,MAAM,EAAE,CAAC;IACrB,YAAY;IACZ,YAAY,EAAE,MAAM,CAAC;IACrB,cAAc;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,mBAAmB;IACnB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,SAAS;IACT,MAAM,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,aAAa;IAC1B,YAAY;IACZ,UAAU,EAAE,MAAM,CAAC;IACnB,aAAa;IACb,QAAQ,EAAE,MAAM,CAAC;IACjB,oBAAoB;IACpB,KAAK,EAAE,KAAK,CAAC;QACT,SAAS,EAAE,MAAM,CAAC;QAClB,QAAQ,EAAE,MAAM,CAAC;QACjB,OAAO,EAAE,MAAM,CAAC;QAChB,UAAU,EAAE,MAAM,CAAC;KACtB,CAAC,CAAC;IACH,oBAAoB;IACpB,SAAS,EAAE,MAAM,EAAE,CAAC;IACpB,uBAAuB;IACvB,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,cAAc;IACd,gBAAgB,EAAE,MAAM,EAAE,CAAC;CAC9B;AAOD,qBAAa,aAAa;IACtB,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,OAAO,CAAiB;gBAEpB,UAAU,EAAE,MAAM;IAM9B;;OAEG;IACH,MAAM,CAAC,MAAM,EAAE,UAAU,EAAE,SAAS,EAAE,KAAK,EAAE,GAAG,IAAI;IA2CpD;;OAEG;IACH,OAAO,CAAC,MAAM,EAAE,MAAM,EAAE,iBAAiB,EAAE,MAAM,EAAE,GAAG,aAAa;IA2CnE,eAAe;IACf,UAAU,IAAI,YAAY,EAAE;IAI5B,WAAW;IACX,KAAK,IAAI,IAAI;IAKb,OAAO,CAAC,WAAW;IAYnB,OAAO,CAAC,IAAI;CAUf"}
@@ -0,0 +1,144 @@
1
+ "use strict";
2
+ /**
3
+ * History Report — 扫描历史记录与趋势分析
4
+ *
5
+ * 功能:
6
+ * 1. 每次扫描后自动记录结果
7
+ * 2. 对比历史数据,高亮新问题 / 已修复问题
8
+ * 3. 生成趋势图表数据(JSON)
9
+ * 4. 团队治理追踪
10
+ */
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.HistoryReport = void 0;
13
+ const node_fs_1 = require("node:fs");
14
+ const node_path_1 = require("node:path");
15
+ /** 问题签名:file|ruleId|line 的哈希,用于跨扫描对比 */
16
+ function issueSignature(issue) {
17
+ return `${issue.file}|${issue.ruleId}|${issue.line}`;
18
+ }
19
+ class HistoryReport {
20
+ historyDir;
21
+ historyFile;
22
+ entries;
23
+ constructor(projectDir) {
24
+ this.historyDir = (0, node_path_1.resolve)(projectDir, ".frontend-guardian");
25
+ this.historyFile = (0, node_path_1.resolve)(this.historyDir, "history.json");
26
+ this.entries = this.loadEntries();
27
+ }
28
+ /**
29
+ * 记录一次扫描结果
30
+ */
31
+ record(result, allIssues) {
32
+ const signatures = allIssues.map(issueSignature);
33
+ const entry = {
34
+ timestamp: Date.now(),
35
+ module: result.module,
36
+ counts: {
37
+ critical: result.issues.critical.length,
38
+ warning: result.issues.warning.length,
39
+ suggestion: result.issues.suggestion.length,
40
+ },
41
+ signatures,
42
+ filesScanned: result.filesScanned,
43
+ duration: result.duration,
44
+ };
45
+ // 尝试获取 git 信息
46
+ try {
47
+ const { execSync } = require("node:child_process");
48
+ entry.commit = execSync("git rev-parse --short HEAD", {
49
+ cwd: this.historyDir.replace("/.frontend-guardian", ""),
50
+ encoding: "utf-8",
51
+ stdio: ["pipe", "pipe", "ignore"],
52
+ }).trim();
53
+ entry.branch = execSync("git branch --show-current", {
54
+ cwd: this.historyDir.replace("/.frontend-guardian", ""),
55
+ encoding: "utf-8",
56
+ stdio: ["pipe", "pipe", "ignore"],
57
+ }).trim();
58
+ }
59
+ catch {
60
+ // 非 git 项目,忽略
61
+ }
62
+ this.entries.push(entry);
63
+ // 只保留最近 100 条记录
64
+ if (this.entries.length > 100) {
65
+ this.entries = this.entries.slice(-100);
66
+ }
67
+ this.save();
68
+ }
69
+ /**
70
+ * 分析趋势:对比本次与上一次同模块扫描
71
+ */
72
+ analyze(module, currentSignatures) {
73
+ const moduleEntries = this.entries.filter((e) => e.module === module);
74
+ const totalScans = moduleEntries.length;
75
+ if (totalScans < 2) {
76
+ return {
77
+ totalScans,
78
+ lastScan: moduleEntries[moduleEntries.length - 1]?.timestamp || Date.now(),
79
+ trend: moduleEntries.map((e) => ({
80
+ timestamp: e.timestamp,
81
+ critical: e.counts.critical,
82
+ warning: e.counts.warning,
83
+ suggestion: e.counts.suggestion,
84
+ })),
85
+ newIssues: [],
86
+ fixedIssues: [],
87
+ persistentIssues: currentSignatures,
88
+ };
89
+ }
90
+ const previous = moduleEntries[moduleEntries.length - 2];
91
+ const prevSet = new Set(previous.signatures);
92
+ const currSet = new Set(currentSignatures);
93
+ const newIssues = currentSignatures.filter((s) => !prevSet.has(s));
94
+ const fixedIssues = previous.signatures.filter((s) => !currSet.has(s));
95
+ const persistentIssues = currentSignatures.filter((s) => prevSet.has(s));
96
+ return {
97
+ totalScans,
98
+ lastScan: previous.timestamp,
99
+ trend: moduleEntries.map((e) => ({
100
+ timestamp: e.timestamp,
101
+ critical: e.counts.critical,
102
+ warning: e.counts.warning,
103
+ suggestion: e.counts.suggestion,
104
+ })),
105
+ newIssues,
106
+ fixedIssues,
107
+ persistentIssues,
108
+ };
109
+ }
110
+ /** 获取所有历史记录 */
111
+ getEntries() {
112
+ return [...this.entries];
113
+ }
114
+ /** 清空历史 */
115
+ clear() {
116
+ this.entries = [];
117
+ this.save();
118
+ }
119
+ loadEntries() {
120
+ try {
121
+ if ((0, node_fs_1.existsSync)(this.historyFile)) {
122
+ const raw = (0, node_fs_1.readFileSync)(this.historyFile, "utf-8");
123
+ return JSON.parse(raw);
124
+ }
125
+ }
126
+ catch {
127
+ // 读取失败返回空
128
+ }
129
+ return [];
130
+ }
131
+ save() {
132
+ try {
133
+ if (!(0, node_fs_1.existsSync)(this.historyDir)) {
134
+ (0, node_fs_1.mkdirSync)(this.historyDir, { recursive: true });
135
+ }
136
+ (0, node_fs_1.writeFileSync)(this.historyFile, JSON.stringify(this.entries, null, 2), "utf-8");
137
+ }
138
+ catch {
139
+ // 静默失败
140
+ }
141
+ }
142
+ }
143
+ exports.HistoryReport = HistoryReport;
144
+ //# sourceMappingURL=history-report.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"history-report.js","sourceRoot":"","sources":["../../src/utils/history-report.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,23 @@
1
+ /**
2
+ * 配置文件初始化工具
3
+ * 一键生成 .frontend-guardian.yml 智能默认配置
4
+ */
5
+ import type { ProjectMeta } from "../types.js";
6
+ /**
7
+ * 生成默认配置 YAML 内容
8
+ * 基于项目检测结果提供智能默认值
9
+ */
10
+ export declare function generateDefaultConfig(meta?: ProjectMeta): string;
11
+ /**
12
+ * 初始化配置文件
13
+ * @param projectDir 项目根目录
14
+ * @param meta 项目元数据(可选,用于智能默认值)
15
+ * @param force 是否覆盖已存在的配置文件
16
+ * @returns 操作结果
17
+ */
18
+ export declare function initConfig(projectDir: string, meta?: ProjectMeta, force?: boolean): {
19
+ created: boolean;
20
+ path: string;
21
+ existed: boolean;
22
+ };
23
+ //# sourceMappingURL=init-config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init-config.d.ts","sourceRoot":"","sources":["../../src/utils/init-config.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAIH,OAAO,KAAK,EAAE,WAAW,EAAuB,MAAM,YAAY,CAAC;AAEnE;;;GAGG;AACH,wBAAgB,qBAAqB,CAAC,IAAI,CAAC,EAAE,WAAW,GAAG,MAAM,CAmNhE;AAED;;;;;;GAMG;AACH,wBAAgB,UAAU,CACtB,UAAU,EAAE,MAAM,EAClB,IAAI,CAAC,EAAE,WAAW,EAClB,KAAK,UAAQ,GACd;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,OAAO,CAAA;CAAE,CAYtD"}