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.
- package/LICENSE +21 -0
- package/bin/fg-core.js +1238 -0
- package/bin/watch-mode.js +123 -0
- package/dist/engine/cache.d.ts +68 -0
- package/dist/engine/cache.d.ts.map +1 -0
- package/dist/engine/cache.js +164 -0
- package/dist/engine/cache.js.map +1 -0
- package/dist/engine/rule-engine.d.ts +135 -0
- package/dist/engine/rule-engine.d.ts.map +1 -0
- package/dist/engine/rule-engine.js +716 -0
- package/dist/engine/rule-engine.js.map +1 -0
- package/dist/formatters/github-annotation.d.ts +36 -0
- package/dist/formatters/github-annotation.d.ts.map +1 -0
- package/dist/formatters/github-annotation.js +122 -0
- package/dist/formatters/github-annotation.js.map +1 -0
- package/dist/formatters/pr-comment.d.ts +43 -0
- package/dist/formatters/pr-comment.d.ts.map +1 -0
- package/dist/formatters/pr-comment.js +171 -0
- package/dist/formatters/pr-comment.js.map +1 -0
- package/dist/formatters/sarif.d.ts +104 -0
- package/dist/formatters/sarif.d.ts.map +1 -0
- package/dist/formatters/sarif.js +130 -0
- package/dist/formatters/sarif.js.map +1 -0
- package/dist/index.d.ts +46 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +108 -0
- package/dist/index.js.map +1 -0
- package/dist/integrations/base.d.ts +44 -0
- package/dist/integrations/base.d.ts.map +1 -0
- package/dist/integrations/base.js +104 -0
- package/dist/integrations/base.js.map +1 -0
- package/dist/integrations/eslint.d.ts +8 -0
- package/dist/integrations/eslint.d.ts.map +1 -0
- package/dist/integrations/eslint.js +67 -0
- package/dist/integrations/eslint.js.map +1 -0
- package/dist/integrations/formatter.d.ts +35 -0
- package/dist/integrations/formatter.d.ts.map +1 -0
- package/dist/integrations/formatter.js +182 -0
- package/dist/integrations/formatter.js.map +1 -0
- package/dist/integrations/index.d.ts +17 -0
- package/dist/integrations/index.d.ts.map +1 -0
- package/dist/integrations/index.js +25 -0
- package/dist/integrations/index.js.map +1 -0
- package/dist/integrations/stylelint.d.ts +8 -0
- package/dist/integrations/stylelint.d.ts.map +1 -0
- package/dist/integrations/stylelint.js +59 -0
- package/dist/integrations/stylelint.js.map +1 -0
- package/dist/integrations/typescript.d.ts +8 -0
- package/dist/integrations/typescript.d.ts.map +1 -0
- package/dist/integrations/typescript.js +92 -0
- package/dist/integrations/typescript.js.map +1 -0
- package/dist/rules/registry.d.ts +83 -0
- package/dist/rules/registry.d.ts.map +1 -0
- package/dist/rules/registry.js +205 -0
- package/dist/rules/registry.js.map +1 -0
- package/dist/scanners/a11y-scanner.d.ts +14 -0
- package/dist/scanners/a11y-scanner.d.ts.map +1 -0
- package/dist/scanners/a11y-scanner.js +781 -0
- package/dist/scanners/a11y-scanner.js.map +1 -0
- package/dist/scanners/component-scanner.d.ts +12 -0
- package/dist/scanners/component-scanner.d.ts.map +1 -0
- package/dist/scanners/component-scanner.js +304 -0
- package/dist/scanners/component-scanner.js.map +1 -0
- package/dist/scanners/cross-file-scanner.d.ts +18 -0
- package/dist/scanners/cross-file-scanner.d.ts.map +1 -0
- package/dist/scanners/cross-file-scanner.js +684 -0
- package/dist/scanners/cross-file-scanner.js.map +1 -0
- package/dist/scanners/hooks-scanner.d.ts +15 -0
- package/dist/scanners/hooks-scanner.d.ts.map +1 -0
- package/dist/scanners/hooks-scanner.js +670 -0
- package/dist/scanners/hooks-scanner.js.map +1 -0
- package/dist/scanners/i18n-scanner.d.ts +13 -0
- package/dist/scanners/i18n-scanner.d.ts.map +1 -0
- package/dist/scanners/i18n-scanner.js +535 -0
- package/dist/scanners/i18n-scanner.js.map +1 -0
- package/dist/scanners/naming-scanner.d.ts +19 -0
- package/dist/scanners/naming-scanner.d.ts.map +1 -0
- package/dist/scanners/naming-scanner.js +746 -0
- package/dist/scanners/naming-scanner.js.map +1 -0
- package/dist/scanners/performance-scanner.d.ts +7 -0
- package/dist/scanners/performance-scanner.d.ts.map +1 -0
- package/dist/scanners/performance-scanner.js +402 -0
- package/dist/scanners/performance-scanner.js.map +1 -0
- package/dist/scanners/platform-scanner.d.ts +15 -0
- package/dist/scanners/platform-scanner.d.ts.map +1 -0
- package/dist/scanners/platform-scanner.js +320 -0
- package/dist/scanners/platform-scanner.js.map +1 -0
- package/dist/scanners/security-scanner.d.ts +7 -0
- package/dist/scanners/security-scanner.d.ts.map +1 -0
- package/dist/scanners/security-scanner.js +349 -0
- package/dist/scanners/security-scanner.js.map +1 -0
- package/dist/scanners/svelte-scanner.d.ts +14 -0
- package/dist/scanners/svelte-scanner.d.ts.map +1 -0
- package/dist/scanners/svelte-scanner.js +228 -0
- package/dist/scanners/svelte-scanner.js.map +1 -0
- package/dist/types.d.ts +343 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +6 -0
- package/dist/types.js.map +1 -0
- package/dist/utils/ast-parser.d.ts +21 -0
- package/dist/utils/ast-parser.d.ts.map +1 -0
- package/dist/utils/ast-parser.js +119 -0
- package/dist/utils/ast-parser.js.map +1 -0
- package/dist/utils/baseline.d.ts +89 -0
- package/dist/utils/baseline.d.ts.map +1 -0
- package/dist/utils/baseline.js +156 -0
- package/dist/utils/baseline.js.map +1 -0
- package/dist/utils/ci-generator.d.ts +34 -0
- package/dist/utils/ci-generator.d.ts.map +1 -0
- package/dist/utils/ci-generator.js +194 -0
- package/dist/utils/ci-generator.js.map +1 -0
- package/dist/utils/common.d.ts +8 -0
- package/dist/utils/common.d.ts.map +1 -0
- package/dist/utils/common.js +38 -0
- package/dist/utils/common.js.map +1 -0
- package/dist/utils/concurrent.d.ts +16 -0
- package/dist/utils/concurrent.d.ts.map +1 -0
- package/dist/utils/concurrent.js +49 -0
- package/dist/utils/concurrent.js.map +1 -0
- package/dist/utils/config-loader.d.ts +8 -0
- package/dist/utils/config-loader.d.ts.map +1 -0
- package/dist/utils/config-loader.js +154 -0
- package/dist/utils/config-loader.js.map +1 -0
- package/dist/utils/fix-bot.d.ts +36 -0
- package/dist/utils/fix-bot.d.ts.map +1 -0
- package/dist/utils/fix-bot.js +274 -0
- package/dist/utils/fix-bot.js.map +1 -0
- package/dist/utils/git-hooks.d.ts +55 -0
- package/dist/utils/git-hooks.d.ts.map +1 -0
- package/dist/utils/git-hooks.js +318 -0
- package/dist/utils/git-hooks.js.map +1 -0
- package/dist/utils/history-report.d.ts +72 -0
- package/dist/utils/history-report.d.ts.map +1 -0
- package/dist/utils/history-report.js +144 -0
- package/dist/utils/history-report.js.map +1 -0
- package/dist/utils/init-config.d.ts +23 -0
- package/dist/utils/init-config.d.ts.map +1 -0
- package/dist/utils/init-config.js +146 -0
- package/dist/utils/init-config.js.map +1 -0
- package/dist/utils/pr-publisher.d.ts +64 -0
- package/dist/utils/pr-publisher.d.ts.map +1 -0
- package/dist/utils/pr-publisher.js +265 -0
- package/dist/utils/pr-publisher.js.map +1 -0
- package/dist/utils/project-detector.d.ts +20 -0
- package/dist/utils/project-detector.d.ts.map +1 -0
- package/dist/utils/project-detector.js +342 -0
- package/dist/utils/project-detector.js.map +1 -0
- package/dist/utils/report-uploader.d.ts +35 -0
- package/dist/utils/report-uploader.d.ts.map +1 -0
- package/dist/utils/report-uploader.js +106 -0
- package/dist/utils/report-uploader.js.map +1 -0
- 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"}
|