@hddz/plugin-harness 0.1.19 → 0.2.1
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.
Potentially problematic release.
This version of @hddz/plugin-harness might be problematic. Click here for more details.
- package/README.md +97 -114
- package/dist/core/auditors/index.d.ts +2 -0
- package/dist/core/auditors/index.js +7 -0
- package/dist/core/auditors/skill-auditor.d.ts +72 -0
- package/dist/core/auditors/skill-auditor.js +488 -0
- package/dist/core/index.d.ts +22 -0
- package/dist/core/index.js +47 -0
- package/dist/core/loggers/config-logger.d.ts +25 -0
- package/dist/core/loggers/config-logger.js +139 -0
- package/dist/core/loggers/index.d.ts +4 -0
- package/dist/core/loggers/index.js +9 -0
- package/dist/core/loggers/operation-logger.d.ts +23 -0
- package/dist/core/loggers/operation-logger.js +125 -0
- package/dist/core/middleware/context-injector.d.ts +25 -0
- package/dist/core/middleware/context-injector.js +174 -0
- package/dist/core/middleware/index.d.ts +5 -0
- package/dist/core/middleware/index.js +11 -0
- package/dist/core/middleware/loop-detector.d.ts +18 -0
- package/dist/core/middleware/loop-detector.js +125 -0
- package/dist/core/middleware/trace-logger.d.ts +34 -0
- package/dist/core/middleware/trace-logger.js +141 -0
- package/dist/core/utils/file.d.ts +28 -0
- package/dist/core/utils/file.js +104 -0
- package/dist/core/utils/format.d.ts +16 -0
- package/dist/core/utils/format.js +60 -0
- package/dist/core/utils/index.d.ts +2 -0
- package/dist/core/utils/index.js +14 -0
- package/dist/core/validators/config-validator.d.ts +25 -0
- package/dist/core/validators/config-validator.js +235 -0
- package/dist/core/validators/index.d.ts +2 -0
- package/dist/core/validators/index.js +7 -0
- package/dist/file-watcher.d.ts +37 -0
- package/dist/file-watcher.js +151 -0
- package/dist/index.d.ts +63 -0
- package/dist/index.js +166 -106
- package/dist/src/file-watcher.d.ts +37 -0
- package/dist/src/file-watcher.js +151 -0
- package/dist/src/index.d.ts +70 -0
- package/dist/src/index.js +192 -0
- package/package.json +4 -12
- package/openclaw.plugin.json +0 -39
|
@@ -0,0 +1,488 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// src/auditors/skill-auditor.ts
|
|
3
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
4
|
+
if (k2 === undefined) k2 = k;
|
|
5
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
6
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
7
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
8
|
+
}
|
|
9
|
+
Object.defineProperty(o, k2, desc);
|
|
10
|
+
}) : (function(o, m, k, k2) {
|
|
11
|
+
if (k2 === undefined) k2 = k;
|
|
12
|
+
o[k2] = m[k];
|
|
13
|
+
}));
|
|
14
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
15
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
16
|
+
}) : function(o, v) {
|
|
17
|
+
o["default"] = v;
|
|
18
|
+
});
|
|
19
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
20
|
+
var ownKeys = function(o) {
|
|
21
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
22
|
+
var ar = [];
|
|
23
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
24
|
+
return ar;
|
|
25
|
+
};
|
|
26
|
+
return ownKeys(o);
|
|
27
|
+
};
|
|
28
|
+
return function (mod) {
|
|
29
|
+
if (mod && mod.__esModule) return mod;
|
|
30
|
+
var result = {};
|
|
31
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
32
|
+
__setModuleDefault(result, mod);
|
|
33
|
+
return result;
|
|
34
|
+
};
|
|
35
|
+
})();
|
|
36
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
37
|
+
exports.SkillAuditor = void 0;
|
|
38
|
+
const fs = __importStar(require("fs-extra"));
|
|
39
|
+
const path = __importStar(require("path"));
|
|
40
|
+
const child_process_1 = require("child_process");
|
|
41
|
+
class SkillAuditor {
|
|
42
|
+
workspacePath;
|
|
43
|
+
constructor(workspacePath) {
|
|
44
|
+
this.workspacePath = workspacePath;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* 审核 Skill
|
|
48
|
+
*/
|
|
49
|
+
async audit(skillPath, source = 'unknown') {
|
|
50
|
+
const warnings = [];
|
|
51
|
+
const recommendations = [];
|
|
52
|
+
// 获取元数据
|
|
53
|
+
const metadata = await this.getSkillMetadata(skillPath);
|
|
54
|
+
// 1. 检查来源
|
|
55
|
+
const sourceWarning = this.checkSource(source);
|
|
56
|
+
if (sourceWarning)
|
|
57
|
+
warnings.push(sourceWarning);
|
|
58
|
+
// 2. 检查 SKILL.md
|
|
59
|
+
if (!metadata.hasSkillMd) {
|
|
60
|
+
warnings.push({
|
|
61
|
+
type: 'permission',
|
|
62
|
+
severity: 'high',
|
|
63
|
+
message: '缺少 SKILL.md 文件,无法确认权限声明',
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
else {
|
|
67
|
+
// 检查权限声明
|
|
68
|
+
const permissionWarning = this.checkPermissions(metadata.declaredPermissions);
|
|
69
|
+
if (permissionWarning)
|
|
70
|
+
warnings.push(permissionWarning);
|
|
71
|
+
}
|
|
72
|
+
// 3. 扫描危险代码
|
|
73
|
+
const codeWarnings = await this.scanDangerousCode(skillPath);
|
|
74
|
+
warnings.push(...codeWarnings);
|
|
75
|
+
// 4. 检查依赖
|
|
76
|
+
if (metadata.hasPackageJson) {
|
|
77
|
+
const depWarnings = await this.checkDependencies(skillPath, metadata.dependencies);
|
|
78
|
+
warnings.push(...depWarnings);
|
|
79
|
+
}
|
|
80
|
+
// 5. 检查脚本目录
|
|
81
|
+
if (metadata.hasScripts) {
|
|
82
|
+
const scriptWarnings = await this.scanScripts(skillPath);
|
|
83
|
+
warnings.push(...scriptWarnings);
|
|
84
|
+
}
|
|
85
|
+
// 计算风险等级
|
|
86
|
+
const riskLevel = this.calculateRiskLevel(warnings);
|
|
87
|
+
const passed = riskLevel !== 'CRITICAL' && riskLevel !== 'HIGH';
|
|
88
|
+
// 生成建议
|
|
89
|
+
if (!passed) {
|
|
90
|
+
recommendations.push('⚠️ 建议人工审核后再决定是否安装');
|
|
91
|
+
}
|
|
92
|
+
if (metadata.dependencies.length > 10) {
|
|
93
|
+
recommendations.push(`📦 依赖较多(${metadata.dependencies.length} 个),建议检查是否有不必要的依赖`);
|
|
94
|
+
}
|
|
95
|
+
if (!metadata.hasSkillMd) {
|
|
96
|
+
recommendations.push('📝 要求作者添加 SKILL.md 权限声明');
|
|
97
|
+
}
|
|
98
|
+
return {
|
|
99
|
+
skillName: path.basename(skillPath),
|
|
100
|
+
skillPath,
|
|
101
|
+
source,
|
|
102
|
+
riskLevel,
|
|
103
|
+
passed,
|
|
104
|
+
warnings,
|
|
105
|
+
recommendations,
|
|
106
|
+
metadata,
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* 检查来源
|
|
111
|
+
*/
|
|
112
|
+
checkSource(source) {
|
|
113
|
+
const trustedSources = [
|
|
114
|
+
'clawhub',
|
|
115
|
+
'official',
|
|
116
|
+
'github.com/openclaw',
|
|
117
|
+
'github.com/clawd',
|
|
118
|
+
];
|
|
119
|
+
const unknownSources = ['unknown', '', undefined];
|
|
120
|
+
if (unknownSources.includes(source)) {
|
|
121
|
+
return {
|
|
122
|
+
type: 'other',
|
|
123
|
+
severity: 'medium',
|
|
124
|
+
message: `来源未知:${source}`,
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
if (trustedSources.some(s => source.toLowerCase().includes(s))) {
|
|
128
|
+
return null; // 可信来源,无警告
|
|
129
|
+
}
|
|
130
|
+
return {
|
|
131
|
+
type: 'other',
|
|
132
|
+
severity: 'low',
|
|
133
|
+
message: `来源:${source}(非官方来源)`,
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
/**
|
|
137
|
+
* 检查权限声明
|
|
138
|
+
*/
|
|
139
|
+
checkPermissions(permissions) {
|
|
140
|
+
const dangerousPermissions = [
|
|
141
|
+
'exec',
|
|
142
|
+
'system',
|
|
143
|
+
'fullFileSystem',
|
|
144
|
+
'network',
|
|
145
|
+
'externalApi',
|
|
146
|
+
];
|
|
147
|
+
const hasDangerous = permissions.some(p => dangerousPermissions.some(dp => p.toLowerCase().includes(dp)));
|
|
148
|
+
if (hasDangerous) {
|
|
149
|
+
return {
|
|
150
|
+
type: 'permission',
|
|
151
|
+
severity: 'medium',
|
|
152
|
+
message: `声明了危险权限:${permissions.filter(p => dangerousPermissions.some(dp => p.toLowerCase().includes(dp))).join(', ')}`,
|
|
153
|
+
};
|
|
154
|
+
}
|
|
155
|
+
if (permissions.length === 0) {
|
|
156
|
+
return {
|
|
157
|
+
type: 'permission',
|
|
158
|
+
severity: 'low',
|
|
159
|
+
message: '未声明任何权限(可能不需要权限,也可能未声明)',
|
|
160
|
+
};
|
|
161
|
+
}
|
|
162
|
+
return null;
|
|
163
|
+
}
|
|
164
|
+
/**
|
|
165
|
+
* 扫描危险代码
|
|
166
|
+
*/
|
|
167
|
+
async scanDangerousCode(skillPath) {
|
|
168
|
+
const warnings = [];
|
|
169
|
+
const dangerousPatterns = [
|
|
170
|
+
{
|
|
171
|
+
pattern: /exec\s*\(/g,
|
|
172
|
+
name: 'exec()',
|
|
173
|
+
severity: 'high',
|
|
174
|
+
message: '发现系统命令执行',
|
|
175
|
+
},
|
|
176
|
+
{
|
|
177
|
+
pattern: /eval\s*\(/g,
|
|
178
|
+
name: 'eval()',
|
|
179
|
+
severity: 'critical',
|
|
180
|
+
message: '发现代码执行(eval)',
|
|
181
|
+
},
|
|
182
|
+
{
|
|
183
|
+
pattern: /child_process/g,
|
|
184
|
+
name: 'child_process',
|
|
185
|
+
severity: 'high',
|
|
186
|
+
message: '使用 Node.js 子进程模块',
|
|
187
|
+
},
|
|
188
|
+
{
|
|
189
|
+
pattern: /spawn\s*\(/g,
|
|
190
|
+
name: 'spawn()',
|
|
191
|
+
severity: 'high',
|
|
192
|
+
message: '发现进程生成',
|
|
193
|
+
},
|
|
194
|
+
{
|
|
195
|
+
pattern: /fs\.writeFile/g,
|
|
196
|
+
name: 'fs.writeFile',
|
|
197
|
+
severity: 'medium',
|
|
198
|
+
message: '发现文件写入操作',
|
|
199
|
+
},
|
|
200
|
+
{
|
|
201
|
+
pattern: /fs\.unlink/g,
|
|
202
|
+
name: 'fs.unlink',
|
|
203
|
+
severity: 'medium',
|
|
204
|
+
message: '发现文件删除操作',
|
|
205
|
+
},
|
|
206
|
+
{
|
|
207
|
+
pattern: /fetch\s*\(/g,
|
|
208
|
+
name: 'fetch()',
|
|
209
|
+
severity: 'medium',
|
|
210
|
+
message: '发现网络请求',
|
|
211
|
+
},
|
|
212
|
+
{
|
|
213
|
+
pattern: /axios\./g,
|
|
214
|
+
name: 'axios',
|
|
215
|
+
severity: 'medium',
|
|
216
|
+
message: '使用 axios 进行网络请求',
|
|
217
|
+
},
|
|
218
|
+
{
|
|
219
|
+
pattern: /https?:\/\/[^\s'"]+/g,
|
|
220
|
+
name: 'hardcoded_url',
|
|
221
|
+
severity: 'low',
|
|
222
|
+
message: '发现硬编码 URL',
|
|
223
|
+
},
|
|
224
|
+
{
|
|
225
|
+
pattern: /process\.env/g,
|
|
226
|
+
name: 'process.env',
|
|
227
|
+
severity: 'low',
|
|
228
|
+
message: '访问环境变量',
|
|
229
|
+
},
|
|
230
|
+
];
|
|
231
|
+
const filesToScan = await this.getJsFiles(skillPath);
|
|
232
|
+
for (const file of filesToScan) {
|
|
233
|
+
try {
|
|
234
|
+
const content = await fs.readFile(file, 'utf8');
|
|
235
|
+
const lines = content.split('\n');
|
|
236
|
+
for (const { pattern, name, severity, message } of dangerousPatterns) {
|
|
237
|
+
pattern.lastIndex = 0; // 重置正则状态
|
|
238
|
+
let match;
|
|
239
|
+
while ((match = pattern.exec(content)) !== null) {
|
|
240
|
+
// 计算行号
|
|
241
|
+
const beforeMatch = content.substring(0, match.index);
|
|
242
|
+
const lineNumber = beforeMatch.split('\n').length;
|
|
243
|
+
// 跳过注释中的匹配
|
|
244
|
+
const line = lines[lineNumber - 1];
|
|
245
|
+
if (line.trim().startsWith('//') || line.trim().startsWith('*')) {
|
|
246
|
+
continue;
|
|
247
|
+
}
|
|
248
|
+
warnings.push({
|
|
249
|
+
type: 'dangerous_code',
|
|
250
|
+
severity,
|
|
251
|
+
message: `${message}: ${name}`,
|
|
252
|
+
file: path.relative(skillPath, file),
|
|
253
|
+
line: lineNumber,
|
|
254
|
+
});
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
catch (error) {
|
|
259
|
+
// 忽略读取错误
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
return warnings;
|
|
263
|
+
}
|
|
264
|
+
/**
|
|
265
|
+
* 扫描脚本文件
|
|
266
|
+
*/
|
|
267
|
+
async scanScripts(skillPath) {
|
|
268
|
+
const warnings = [];
|
|
269
|
+
const scriptsDir = path.join(skillPath, 'scripts');
|
|
270
|
+
if (!await fs.pathExists(scriptsDir)) {
|
|
271
|
+
return warnings;
|
|
272
|
+
}
|
|
273
|
+
const scriptFiles = await fs.readdir(scriptsDir);
|
|
274
|
+
for (const script of scriptFiles) {
|
|
275
|
+
const scriptPath = path.join(scriptsDir, script);
|
|
276
|
+
// 检查 shell 脚本
|
|
277
|
+
if (script.endsWith('.sh')) {
|
|
278
|
+
const content = await fs.readFile(scriptPath, 'utf8');
|
|
279
|
+
const dangerousCommands = [
|
|
280
|
+
{ pattern: /rm\s+-rf/g, message: '发现危险命令:rm -rf' },
|
|
281
|
+
{ pattern: /curl.*\|\s*bash/g, message: '发现管道执行:curl | bash' },
|
|
282
|
+
{ pattern: /wget.*\|\s*bash/g, message: '发现管道执行:wget | bash' },
|
|
283
|
+
{ pattern: /sudo/g, message: '使用 sudo 提权' },
|
|
284
|
+
{ pattern: /chmod\s+777/g, message: '设置文件权限为 777' },
|
|
285
|
+
];
|
|
286
|
+
for (const { pattern, message } of dangerousCommands) {
|
|
287
|
+
if (pattern.test(content)) {
|
|
288
|
+
warnings.push({
|
|
289
|
+
type: 'dangerous_code',
|
|
290
|
+
severity: 'high',
|
|
291
|
+
message,
|
|
292
|
+
file: `scripts/${script}`,
|
|
293
|
+
});
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
return warnings;
|
|
299
|
+
}
|
|
300
|
+
/**
|
|
301
|
+
* 检查依赖
|
|
302
|
+
*/
|
|
303
|
+
async checkDependencies(skillPath, dependencies) {
|
|
304
|
+
const warnings = [];
|
|
305
|
+
// 检查依赖数量
|
|
306
|
+
if (dependencies.length > 20) {
|
|
307
|
+
warnings.push({
|
|
308
|
+
type: 'dependency',
|
|
309
|
+
severity: 'low',
|
|
310
|
+
message: `依赖过多(${dependencies.length} 个),可能增加攻击面`,
|
|
311
|
+
});
|
|
312
|
+
}
|
|
313
|
+
// 尝试运行 npm audit
|
|
314
|
+
try {
|
|
315
|
+
const auditOutput = (0, child_process_1.execSync)('npm audit --json', {
|
|
316
|
+
cwd: skillPath,
|
|
317
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
318
|
+
encoding: 'utf8',
|
|
319
|
+
});
|
|
320
|
+
const audit = JSON.parse(auditOutput);
|
|
321
|
+
const vulnerabilities = audit.metadata?.vulnerabilities;
|
|
322
|
+
if (vulnerabilities && vulnerabilities.total > 0) {
|
|
323
|
+
warnings.push({
|
|
324
|
+
type: 'dependency',
|
|
325
|
+
severity: vulnerabilities.critical > 0 || vulnerabilities.high > 0 ? 'high' : 'medium',
|
|
326
|
+
message: `发现 ${vulnerabilities.total} 个依赖漏洞(${vulnerabilities.critical} 严重,${vulnerabilities.high} 高)`,
|
|
327
|
+
});
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
catch (error) {
|
|
331
|
+
// npm audit 可能失败,忽略
|
|
332
|
+
}
|
|
333
|
+
return warnings;
|
|
334
|
+
}
|
|
335
|
+
/**
|
|
336
|
+
* 获取 Skill 元数据
|
|
337
|
+
*/
|
|
338
|
+
async getSkillMetadata(skillPath) {
|
|
339
|
+
const metadata = {
|
|
340
|
+
hasSkillMd: false,
|
|
341
|
+
hasScripts: false,
|
|
342
|
+
hasPackageJson: false,
|
|
343
|
+
declaredPermissions: [],
|
|
344
|
+
dependencies: [],
|
|
345
|
+
devDependencies: [],
|
|
346
|
+
};
|
|
347
|
+
// 检查 SKILL.md
|
|
348
|
+
const skillMdPath = path.join(skillPath, 'SKILL.md');
|
|
349
|
+
if (await fs.pathExists(skillMdPath)) {
|
|
350
|
+
metadata.hasSkillMd = true;
|
|
351
|
+
const content = await fs.readFile(skillMdPath, 'utf8');
|
|
352
|
+
// 提取权限声明(假设格式为 "Permissions: exec, network")
|
|
353
|
+
const permissionMatch = content.match(/Permissions?:\s*([^\n]+)/i);
|
|
354
|
+
if (permissionMatch) {
|
|
355
|
+
metadata.declaredPermissions = permissionMatch[1]
|
|
356
|
+
.split(',')
|
|
357
|
+
.map(p => p.trim())
|
|
358
|
+
.filter(p => p);
|
|
359
|
+
}
|
|
360
|
+
}
|
|
361
|
+
// 检查 scripts 目录
|
|
362
|
+
const scriptsDir = path.join(skillPath, 'scripts');
|
|
363
|
+
metadata.hasScripts = await fs.pathExists(scriptsDir);
|
|
364
|
+
// 检查 package.json
|
|
365
|
+
const packageJsonPath = path.join(skillPath, 'package.json');
|
|
366
|
+
if (await fs.pathExists(packageJsonPath)) {
|
|
367
|
+
metadata.hasPackageJson = true;
|
|
368
|
+
try {
|
|
369
|
+
const pkg = JSON.parse(await fs.readFile(packageJsonPath, 'utf8'));
|
|
370
|
+
metadata.dependencies = Object.keys(pkg.dependencies || {});
|
|
371
|
+
metadata.devDependencies = Object.keys(pkg.devDependencies || {});
|
|
372
|
+
}
|
|
373
|
+
catch (error) {
|
|
374
|
+
// 忽略解析错误
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
// 获取最后修改时间
|
|
378
|
+
try {
|
|
379
|
+
const stats = await fs.stat(skillPath);
|
|
380
|
+
metadata.lastModified = stats.mtime;
|
|
381
|
+
}
|
|
382
|
+
catch (error) {
|
|
383
|
+
// 忽略
|
|
384
|
+
}
|
|
385
|
+
return metadata;
|
|
386
|
+
}
|
|
387
|
+
/**
|
|
388
|
+
* 获取所有 JS 文件
|
|
389
|
+
*/
|
|
390
|
+
async getJsFiles(dir) {
|
|
391
|
+
const files = [];
|
|
392
|
+
async function scan(currentDir) {
|
|
393
|
+
const entries = await fs.readdir(currentDir, { withFileTypes: true });
|
|
394
|
+
for (const entry of entries) {
|
|
395
|
+
const fullPath = path.join(currentDir, entry.name);
|
|
396
|
+
if (entry.isDirectory()) {
|
|
397
|
+
// 跳过 node_modules
|
|
398
|
+
if (entry.name === 'node_modules' || entry.name === 'dist') {
|
|
399
|
+
continue;
|
|
400
|
+
}
|
|
401
|
+
await scan(fullPath);
|
|
402
|
+
}
|
|
403
|
+
else if (entry.isFile() && (entry.name.endsWith('.js') || entry.name.endsWith('.ts'))) {
|
|
404
|
+
files.push(fullPath);
|
|
405
|
+
}
|
|
406
|
+
}
|
|
407
|
+
}
|
|
408
|
+
await scan(dir);
|
|
409
|
+
return files;
|
|
410
|
+
}
|
|
411
|
+
/**
|
|
412
|
+
* 计算风险等级
|
|
413
|
+
*/
|
|
414
|
+
calculateRiskLevel(warnings) {
|
|
415
|
+
const criticalCount = warnings.filter(w => w.severity === 'critical').length;
|
|
416
|
+
const highCount = warnings.filter(w => w.severity === 'high').length;
|
|
417
|
+
const mediumCount = warnings.filter(w => w.severity === 'medium').length;
|
|
418
|
+
if (criticalCount > 0)
|
|
419
|
+
return 'CRITICAL';
|
|
420
|
+
if (highCount > 2)
|
|
421
|
+
return 'HIGH';
|
|
422
|
+
if (highCount > 0 || mediumCount > 3)
|
|
423
|
+
return 'MEDIUM';
|
|
424
|
+
return 'LOW';
|
|
425
|
+
}
|
|
426
|
+
/**
|
|
427
|
+
* 生成审计报告
|
|
428
|
+
*/
|
|
429
|
+
generateReport(result) {
|
|
430
|
+
const lines = [];
|
|
431
|
+
lines.push(`# Skill 安全审计报告`);
|
|
432
|
+
lines.push('');
|
|
433
|
+
lines.push(`## 基本信息`);
|
|
434
|
+
lines.push(`- **Skill 名称**: ${result.skillName}`);
|
|
435
|
+
lines.push(`- **路径**: ${result.skillPath}`);
|
|
436
|
+
lines.push(`- **来源**: ${result.source}`);
|
|
437
|
+
lines.push(`- **风险等级**: ${this.getRiskEmoji(result.riskLevel)} ${result.riskLevel}`);
|
|
438
|
+
lines.push(`- **审核结果**: ${result.passed ? '✅ 通过' : '❌ 未通过'}`);
|
|
439
|
+
lines.push('');
|
|
440
|
+
if (result.warnings.length > 0) {
|
|
441
|
+
lines.push(`## 警告(${result.warnings.length} 个)`);
|
|
442
|
+
lines.push('');
|
|
443
|
+
for (const warning of result.warnings) {
|
|
444
|
+
lines.push(`- ${this.getSeverityEmoji(warning.severity)} **[${warning.type}]** ${warning.message}`);
|
|
445
|
+
if (warning.file) {
|
|
446
|
+
lines.push(` - 文件:\`${warning.file}\`${warning.line ? ` 第 ${warning.line} 行` : ''}`);
|
|
447
|
+
}
|
|
448
|
+
}
|
|
449
|
+
lines.push('');
|
|
450
|
+
}
|
|
451
|
+
if (result.recommendations.length > 0) {
|
|
452
|
+
lines.push(`## 建议`);
|
|
453
|
+
lines.push('');
|
|
454
|
+
for (const rec of result.recommendations) {
|
|
455
|
+
lines.push(`- ${rec}`);
|
|
456
|
+
}
|
|
457
|
+
lines.push('');
|
|
458
|
+
}
|
|
459
|
+
lines.push(`## 元数据`);
|
|
460
|
+
lines.push(`- 有 SKILL.md: ${result.metadata.hasSkillMd ? '✅' : '❌'}`);
|
|
461
|
+
lines.push(`- 有 scripts 目录:${result.metadata.hasScripts ? '✅' : '❌'}`);
|
|
462
|
+
lines.push(`- 有 package.json: ${result.metadata.hasPackageJson ? '✅' : '❌'}`);
|
|
463
|
+
lines.push(`- 声明权限:${result.metadata.declaredPermissions.join(', ') || '无'}`);
|
|
464
|
+
lines.push(`- 依赖数量:${result.metadata.dependencies.length}`);
|
|
465
|
+
lines.push('');
|
|
466
|
+
return lines.join('\n');
|
|
467
|
+
}
|
|
468
|
+
getRiskEmoji(level) {
|
|
469
|
+
const emojis = {
|
|
470
|
+
LOW: '🟢',
|
|
471
|
+
MEDIUM: '🟡',
|
|
472
|
+
HIGH: '🟠',
|
|
473
|
+
CRITICAL: '🔴',
|
|
474
|
+
};
|
|
475
|
+
return emojis[level] || '⚪';
|
|
476
|
+
}
|
|
477
|
+
getSeverityEmoji(severity) {
|
|
478
|
+
const emojis = {
|
|
479
|
+
low: 'ℹ️',
|
|
480
|
+
medium: '⚠️',
|
|
481
|
+
high: '🚨',
|
|
482
|
+
critical: '💀',
|
|
483
|
+
};
|
|
484
|
+
return emojis[severity] || '⚪';
|
|
485
|
+
}
|
|
486
|
+
}
|
|
487
|
+
exports.SkillAuditor = SkillAuditor;
|
|
488
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"skill-auditor.js","sourceRoot":"","sources":["../../../src/core/auditors/skill-auditor.ts"],"names":[],"mappings":"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","sourcesContent":["// src/auditors/skill-auditor.ts\n\nimport * as fs from 'fs-extra';\nimport * as path from 'path';\nimport { execSync } from 'child_process';\n\nexport interface SkillAuditResult {\n  skillName: string;\n  skillPath: string;\n  source: string;\n  riskLevel: 'LOW' | 'MEDIUM' | 'HIGH' | 'CRITICAL';\n  passed: boolean;\n  warnings: Warning[];\n  recommendations: string[];\n  metadata: SkillMetadata;\n}\n\nexport interface Warning {\n  type: 'dangerous_code' | 'permission' | 'dependency' | 'network' | 'other';\n  severity: 'low' | 'medium' | 'high' | 'critical';\n  message: string;\n  file?: string;\n  line?: number;\n}\n\nexport interface SkillMetadata {\n  hasSkillMd: boolean;\n  hasScripts: boolean;\n  hasPackageJson: boolean;\n  declaredPermissions: string[];\n  dependencies: string[];\n  devDependencies: string[];\n  lastModified?: Date;\n}\n\nexport class SkillAuditor {\n  private workspacePath: string;\n\n  constructor(workspacePath: string) {\n    this.workspacePath = workspacePath;\n  }\n\n  /**\n   * 审核 Skill\n   */\n  async audit(skillPath: string, source: string = 'unknown'): Promise<SkillAuditResult> {\n    const warnings: Warning[] = [];\n    const recommendations: string[] = [];\n\n    // 获取元数据\n    const metadata = await this.getSkillMetadata(skillPath);\n\n    // 1. 检查来源\n    const sourceWarning = this.checkSource(source);\n    if (sourceWarning) warnings.push(sourceWarning);\n\n    // 2. 检查 SKILL.md\n    if (!metadata.hasSkillMd) {\n      warnings.push({\n        type: 'permission',\n        severity: 'high',\n        message: '缺少 SKILL.md 文件，无法确认权限声明',\n      });\n    } else {\n      // 检查权限声明\n      const permissionWarning = this.checkPermissions(metadata.declaredPermissions);\n      if (permissionWarning) warnings.push(permissionWarning);\n    }\n\n    // 3. 扫描危险代码\n    const codeWarnings = await this.scanDangerousCode(skillPath);\n    warnings.push(...codeWarnings);\n\n    // 4. 检查依赖\n    if (metadata.hasPackageJson) {\n      const depWarnings = await this.checkDependencies(skillPath, metadata.dependencies);\n      warnings.push(...depWarnings);\n    }\n\n    // 5. 检查脚本目录\n    if (metadata.hasScripts) {\n      const scriptWarnings = await this.scanScripts(skillPath);\n      warnings.push(...scriptWarnings);\n    }\n\n    // 计算风险等级\n    const riskLevel = this.calculateRiskLevel(warnings);\n    const passed = riskLevel !== 'CRITICAL' && riskLevel !== 'HIGH';\n\n    // 生成建议\n    if (!passed) {\n      recommendations.push('⚠️ 建议人工审核后再决定是否安装');\n    }\n    if (metadata.dependencies.length > 10) {\n      recommendations.push(`📦 依赖较多（${metadata.dependencies.length} 个），建议检查是否有不必要的依赖`);\n    }\n    if (!metadata.hasSkillMd) {\n      recommendations.push('📝 要求作者添加 SKILL.md 权限声明');\n    }\n\n    return {\n      skillName: path.basename(skillPath),\n      skillPath,\n      source,\n      riskLevel,\n      passed,\n      warnings,\n      recommendations,\n      metadata,\n    };\n  }\n\n  /**\n   * 检查来源\n   */\n  private checkSource(source: string): Warning | null {\n    const trustedSources = [\n      'clawhub',\n      'official',\n      'github.com/openclaw',\n      'github.com/clawd',\n    ];\n\n    const unknownSources = ['unknown', '', undefined];\n\n    if (unknownSources.includes(source)) {\n      return {\n        type: 'other',\n        severity: 'medium',\n        message: `来源未知：${source}`,\n      };\n    }\n\n    if (trustedSources.some(s => source.toLowerCase().includes(s))) {\n      return null; // 可信来源，无警告\n    }\n\n    return {\n      type: 'other',\n      severity: 'low',\n      message: `来源：${source}（非官方来源）`,\n    };\n  }\n\n  /**\n   * 检查权限声明\n   */\n  private checkPermissions(permissions: string[]): Warning | null {\n    const dangerousPermissions = [\n      'exec',\n      'system',\n      'fullFileSystem',\n      'network',\n      'externalApi',\n    ];\n\n    const hasDangerous = permissions.some(p =>\n      dangerousPermissions.some(dp => p.toLowerCase().includes(dp))\n    );\n\n    if (hasDangerous) {\n      return {\n        type: 'permission',\n        severity: 'medium',\n        message: `声明了危险权限：${permissions.filter(p => dangerousPermissions.some(dp => p.toLowerCase().includes(dp))).join(', ')}`,\n      };\n    }\n\n    if (permissions.length === 0) {\n      return {\n        type: 'permission',\n        severity: 'low',\n        message: '未声明任何权限（可能不需要权限，也可能未声明）',\n      };\n    }\n\n    return null;\n  }\n\n  /**\n   * 扫描危险代码\n   */\n  private async scanDangerousCode(skillPath: string): Promise<Warning[]> {\n    const warnings: Warning[] = [];\n\n    const dangerousPatterns = [\n      {\n        pattern: /exec\\s*\\(/g,\n        name: 'exec()',\n        severity: 'high' as const,\n        message: '发现系统命令执行',\n      },\n      {\n        pattern: /eval\\s*\\(/g,\n        name: 'eval()',\n        severity: 'critical' as const,\n        message: '发现代码执行（eval）',\n      },\n      {\n        pattern: /child_process/g,\n        name: 'child_process',\n        severity: 'high' as const,\n        message: '使用 Node.js 子进程模块',\n      },\n      {\n        pattern: /spawn\\s*\\(/g,\n        name: 'spawn()',\n        severity: 'high' as const,\n        message: '发现进程生成',\n      },\n      {\n        pattern: /fs\\.writeFile/g,\n        name: 'fs.writeFile',\n        severity: 'medium' as const,\n        message: '发现文件写入操作',\n      },\n      {\n        pattern: /fs\\.unlink/g,\n        name: 'fs.unlink',\n        severity: 'medium' as const,\n        message: '发现文件删除操作',\n      },\n      {\n        pattern: /fetch\\s*\\(/g,\n        name: 'fetch()',\n        severity: 'medium' as const,\n        message: '发现网络请求',\n      },\n      {\n        pattern: /axios\\./g,\n        name: 'axios',\n        severity: 'medium' as const,\n        message: '使用 axios 进行网络请求',\n      },\n      {\n        pattern: /https?:\\/\\/[^\\s'\"]+/g,\n        name: 'hardcoded_url',\n        severity: 'low' as const,\n        message: '发现硬编码 URL',\n      },\n      {\n        pattern: /process\\.env/g,\n        name: 'process.env',\n        severity: 'low' as const,\n        message: '访问环境变量',\n      },\n    ];\n\n    const filesToScan = await this.getJsFiles(skillPath);\n\n    for (const file of filesToScan) {\n      try {\n        const content = await fs.readFile(file, 'utf8');\n        const lines = content.split('\\n');\n\n        for (const { pattern, name, severity, message } of dangerousPatterns) {\n          pattern.lastIndex = 0; // 重置正则状态\n          let match;\n          while ((match = pattern.exec(content)) !== null) {\n            // 计算行号\n            const beforeMatch = content.substring(0, match.index);\n            const lineNumber = beforeMatch.split('\\n').length;\n\n            // 跳过注释中的匹配\n            const line = lines[lineNumber - 1];\n            if (line.trim().startsWith('//') || line.trim().startsWith('*')) {\n              continue;\n            }\n\n            warnings.push({\n              type: 'dangerous_code',\n              severity,\n              message: `${message}: ${name}`,\n              file: path.relative(skillPath, file),\n              line: lineNumber,\n            });\n          }\n        }\n      } catch (error) {\n        // 忽略读取错误\n      }\n    }\n\n    return warnings;\n  }\n\n  /**\n   * 扫描脚本文件\n   */\n  private async scanScripts(skillPath: string): Promise<Warning[]> {\n    const warnings: Warning[] = [];\n    const scriptsDir = path.join(skillPath, 'scripts');\n\n    if (!await fs.pathExists(scriptsDir)) {\n      return warnings;\n    }\n\n    const scriptFiles = await fs.readdir(scriptsDir);\n\n    for (const script of scriptFiles) {\n      const scriptPath = path.join(scriptsDir, script);\n\n      // 检查 shell 脚本\n      if (script.endsWith('.sh')) {\n        const content = await fs.readFile(scriptPath, 'utf8');\n\n        const dangerousCommands = [\n          { pattern: /rm\\s+-rf/g, message: '发现危险命令：rm -rf' },\n          { pattern: /curl.*\\|\\s*bash/g, message: '发现管道执行：curl | bash' },\n          { pattern: /wget.*\\|\\s*bash/g, message: '发现管道执行：wget | bash' },\n          { pattern: /sudo/g, message: '使用 sudo 提权' },\n          { pattern: /chmod\\s+777/g, message: '设置文件权限为 777' },\n        ];\n\n        for (const { pattern, message } of dangerousCommands) {\n          if (pattern.test(content)) {\n            warnings.push({\n              type: 'dangerous_code',\n              severity: 'high',\n              message,\n              file: `scripts/${script}`,\n            });\n          }\n        }\n      }\n    }\n\n    return warnings;\n  }\n\n  /**\n   * 检查依赖\n   */\n  private async checkDependencies(skillPath: string, dependencies: string[]): Promise<Warning[]> {\n    const warnings: Warning[] = [];\n\n    // 检查依赖数量\n    if (dependencies.length > 20) {\n      warnings.push({\n        type: 'dependency',\n        severity: 'low',\n        message: `依赖过多（${dependencies.length} 个），可能增加攻击面`,\n      });\n    }\n\n    // 尝试运行 npm audit\n    try {\n      const auditOutput = execSync('npm audit --json', {\n        cwd: skillPath,\n        stdio: ['pipe', 'pipe', 'pipe'],\n        encoding: 'utf8',\n      });\n\n      const audit = JSON.parse(auditOutput);\n      const vulnerabilities = audit.metadata?.vulnerabilities;\n\n      if (vulnerabilities && vulnerabilities.total > 0) {\n        warnings.push({\n          type: 'dependency',\n          severity: vulnerabilities.critical > 0 || vulnerabilities.high > 0 ? 'high' : 'medium',\n          message: `发现 ${vulnerabilities.total} 个依赖漏洞（${vulnerabilities.critical} 严重，${vulnerabilities.high} 高）`,\n        });\n      }\n    } catch (error) {\n      // npm audit 可能失败，忽略\n    }\n\n    return warnings;\n  }\n\n  /**\n   * 获取 Skill 元数据\n   */\n  private async getSkillMetadata(skillPath: string): Promise<SkillMetadata> {\n    const metadata: SkillMetadata = {\n      hasSkillMd: false,\n      hasScripts: false,\n      hasPackageJson: false,\n      declaredPermissions: [],\n      dependencies: [],\n      devDependencies: [],\n    };\n\n    // 检查 SKILL.md\n    const skillMdPath = path.join(skillPath, 'SKILL.md');\n    if (await fs.pathExists(skillMdPath)) {\n      metadata.hasSkillMd = true;\n      const content = await fs.readFile(skillMdPath, 'utf8');\n\n      // 提取权限声明（假设格式为 \"Permissions: exec, network\"）\n      const permissionMatch = content.match(/Permissions?:\\s*([^\\n]+)/i);\n      if (permissionMatch) {\n        metadata.declaredPermissions = permissionMatch[1]\n          .split(',')\n          .map(p => p.trim())\n          .filter(p => p);\n      }\n    }\n\n    // 检查 scripts 目录\n    const scriptsDir = path.join(skillPath, 'scripts');\n    metadata.hasScripts = await fs.pathExists(scriptsDir);\n\n    // 检查 package.json\n    const packageJsonPath = path.join(skillPath, 'package.json');\n    if (await fs.pathExists(packageJsonPath)) {\n      metadata.hasPackageJson = true;\n      try {\n        const pkg = JSON.parse(await fs.readFile(packageJsonPath, 'utf8'));\n        metadata.dependencies = Object.keys(pkg.dependencies || {});\n        metadata.devDependencies = Object.keys(pkg.devDependencies || {});\n      } catch (error) {\n        // 忽略解析错误\n      }\n    }\n\n    // 获取最后修改时间\n    try {\n      const stats = await fs.stat(skillPath);\n      metadata.lastModified = stats.mtime;\n    } catch (error) {\n      // 忽略\n    }\n\n    return metadata;\n  }\n\n  /**\n   * 获取所有 JS 文件\n   */\n  private async getJsFiles(dir: string): Promise<string[]> {\n    const files: string[] = [];\n\n    async function scan(currentDir: string) {\n      const entries = await fs.readdir(currentDir, { withFileTypes: true });\n\n      for (const entry of entries) {\n        const fullPath = path.join(currentDir, entry.name);\n\n        if (entry.isDirectory()) {\n          // 跳过 node_modules\n          if (entry.name === 'node_modules' || entry.name === 'dist') {\n            continue;\n          }\n          await scan(fullPath);\n        } else if (entry.isFile() && (entry.name.endsWith('.js') || entry.name.endsWith('.ts'))) {\n          files.push(fullPath);\n        }\n      }\n    }\n\n    await scan(dir);\n    return files;\n  }\n\n  /**\n   * 计算风险等级\n   */\n  private calculateRiskLevel(warnings: Warning[]): 'LOW' | 'MEDIUM' | 'HIGH' | 'CRITICAL' {\n    const criticalCount = warnings.filter(w => w.severity === 'critical').length;\n    const highCount = warnings.filter(w => w.severity === 'high').length;\n    const mediumCount = warnings.filter(w => w.severity === 'medium').length;\n\n    if (criticalCount > 0) return 'CRITICAL';\n    if (highCount > 2) return 'HIGH';\n    if (highCount > 0 || mediumCount > 3) return 'MEDIUM';\n    return 'LOW';\n  }\n\n  /**\n   * 生成审计报告\n   */\n  generateReport(result: SkillAuditResult): string {\n    const lines: string[] = [];\n\n    lines.push(`# Skill 安全审计报告`);\n    lines.push('');\n    lines.push(`## 基本信息`);\n    lines.push(`- **Skill 名称**: ${result.skillName}`);\n    lines.push(`- **路径**: ${result.skillPath}`);\n    lines.push(`- **来源**: ${result.source}`);\n    lines.push(`- **风险等级**: ${this.getRiskEmoji(result.riskLevel)} ${result.riskLevel}`);\n    lines.push(`- **审核结果**: ${result.passed ? '✅ 通过' : '❌ 未通过'}`);\n    lines.push('');\n\n    if (result.warnings.length > 0) {\n      lines.push(`## 警告（${result.warnings.length} 个）`);\n      lines.push('');\n      for (const warning of result.warnings) {\n        lines.push(`- ${this.getSeverityEmoji(warning.severity)} **[${warning.type}]** ${warning.message}`);\n        if (warning.file) {\n          lines.push(`  - 文件：\\`${warning.file}\\`${warning.line ? ` 第 ${warning.line} 行` : ''}`);\n        }\n      }\n      lines.push('');\n    }\n\n    if (result.recommendations.length > 0) {\n      lines.push(`## 建议`);\n      lines.push('');\n      for (const rec of result.recommendations) {\n        lines.push(`- ${rec}`);\n      }\n      lines.push('');\n    }\n\n    lines.push(`## 元数据`);\n    lines.push(`- 有 SKILL.md: ${result.metadata.hasSkillMd ? '✅' : '❌'}`);\n    lines.push(`- 有 scripts 目录：${result.metadata.hasScripts ? '✅' : '❌'}`);\n    lines.push(`- 有 package.json: ${result.metadata.hasPackageJson ? '✅' : '❌'}`);\n    lines.push(`- 声明权限：${result.metadata.declaredPermissions.join(', ') || '无'}`);\n    lines.push(`- 依赖数量：${result.metadata.dependencies.length}`);\n    lines.push('');\n\n    return lines.join('\\n');\n  }\n\n  private getRiskEmoji(level: string): string {\n    const emojis: Record<string, string> = {\n      LOW: '🟢',\n      MEDIUM: '🟡',\n      HIGH: '🟠',\n      CRITICAL: '🔴',\n    };\n    return emojis[level] || '⚪';\n  }\n\n  private getSeverityEmoji(severity: string): string {\n    const emojis: Record<string, string> = {\n      low: 'ℹ️',\n      medium: '⚠️',\n      high: '🚨',\n      critical: '💀',\n    };\n    return emojis[severity] || '⚪';\n  }\n}\n"]}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
export * from './middleware';
|
|
2
|
+
export * from './validators';
|
|
3
|
+
export * from './loggers';
|
|
4
|
+
export * from './utils';
|
|
5
|
+
export * from './auditors';
|
|
6
|
+
import { LoopDetector } from './middleware/loop-detector';
|
|
7
|
+
import { TraceLogger } from './middleware/trace-logger';
|
|
8
|
+
import { ContextInjector } from './middleware/context-injector';
|
|
9
|
+
import { ConfigValidator } from './validators/config-validator';
|
|
10
|
+
import { ConfigLogger } from './loggers/config-logger';
|
|
11
|
+
import { OperationLogger } from './loggers/operation-logger';
|
|
12
|
+
export interface Harness {
|
|
13
|
+
loopDetector: LoopDetector;
|
|
14
|
+
traceLogger: TraceLogger;
|
|
15
|
+
contextInjector: ContextInjector;
|
|
16
|
+
configValidator: ConfigValidator;
|
|
17
|
+
configLogger: ConfigLogger;
|
|
18
|
+
operationLogger: OperationLogger;
|
|
19
|
+
}
|
|
20
|
+
export declare function createHarness(workspacePath: string): Harness;
|
|
21
|
+
export declare function createConfigValidator(): ConfigValidator;
|
|
22
|
+
export default createHarness;
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// src/index.ts - 主入口
|
|
3
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
4
|
+
if (k2 === undefined) k2 = k;
|
|
5
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
6
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
7
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
8
|
+
}
|
|
9
|
+
Object.defineProperty(o, k2, desc);
|
|
10
|
+
}) : (function(o, m, k, k2) {
|
|
11
|
+
if (k2 === undefined) k2 = k;
|
|
12
|
+
o[k2] = m[k];
|
|
13
|
+
}));
|
|
14
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
15
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
16
|
+
};
|
|
17
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
18
|
+
exports.createHarness = createHarness;
|
|
19
|
+
exports.createConfigValidator = createConfigValidator;
|
|
20
|
+
__exportStar(require("./middleware"), exports);
|
|
21
|
+
__exportStar(require("./validators"), exports);
|
|
22
|
+
__exportStar(require("./loggers"), exports);
|
|
23
|
+
__exportStar(require("./utils"), exports);
|
|
24
|
+
__exportStar(require("./auditors"), exports);
|
|
25
|
+
// 便捷工厂函数
|
|
26
|
+
const loop_detector_1 = require("./middleware/loop-detector");
|
|
27
|
+
const trace_logger_1 = require("./middleware/trace-logger");
|
|
28
|
+
const context_injector_1 = require("./middleware/context-injector");
|
|
29
|
+
const config_validator_1 = require("./validators/config-validator");
|
|
30
|
+
const config_logger_1 = require("./loggers/config-logger");
|
|
31
|
+
const operation_logger_1 = require("./loggers/operation-logger");
|
|
32
|
+
function createHarness(workspacePath) {
|
|
33
|
+
return {
|
|
34
|
+
loopDetector: new loop_detector_1.LoopDetector(workspacePath),
|
|
35
|
+
traceLogger: new trace_logger_1.TraceLogger(workspacePath),
|
|
36
|
+
contextInjector: new context_injector_1.ContextInjector(workspacePath),
|
|
37
|
+
configValidator: new config_validator_1.ConfigValidator(workspacePath),
|
|
38
|
+
configLogger: new config_logger_1.ConfigLogger(workspacePath),
|
|
39
|
+
operationLogger: new operation_logger_1.OperationLogger(workspacePath),
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
// 便捷导出(用于包装层)
|
|
43
|
+
function createConfigValidator() {
|
|
44
|
+
return new config_validator_1.ConfigValidator(process.cwd());
|
|
45
|
+
}
|
|
46
|
+
exports.default = createHarness;
|
|
47
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29yZS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEscUJBQXFCOzs7Ozs7Ozs7Ozs7Ozs7O0FBeUJyQixzQ0FTQztBQUdELHNEQUVDO0FBckNELCtDQUE2QjtBQUM3QiwrQ0FBNkI7QUFDN0IsNENBQTBCO0FBQzFCLDBDQUF3QjtBQUN4Qiw2Q0FBMkI7QUFFM0IsU0FBUztBQUNULDhEQUEwRDtBQUMxRCw0REFBd0Q7QUFDeEQsb0VBQWdFO0FBQ2hFLG9FQUFnRTtBQUNoRSwyREFBdUQ7QUFDdkQsaUVBQTZEO0FBVzdELFNBQWdCLGFBQWEsQ0FBQyxhQUFxQjtJQUNqRCxPQUFPO1FBQ0wsWUFBWSxFQUFFLElBQUksNEJBQVksQ0FBQyxhQUFhLENBQUM7UUFDN0MsV0FBVyxFQUFFLElBQUksMEJBQVcsQ0FBQyxhQUFhLENBQUM7UUFDM0MsZUFBZSxFQUFFLElBQUksa0NBQWUsQ0FBQyxhQUFhLENBQUM7UUFDbkQsZUFBZSxFQUFFLElBQUksa0NBQWUsQ0FBQyxhQUFhLENBQUM7UUFDbkQsWUFBWSxFQUFFLElBQUksNEJBQVksQ0FBQyxhQUFhLENBQUM7UUFDN0MsZUFBZSxFQUFFLElBQUksa0NBQWUsQ0FBQyxhQUFhLENBQUM7S0FDcEQsQ0FBQztBQUNKLENBQUM7QUFFRCxjQUFjO0FBQ2QsU0FBZ0IscUJBQXFCO0lBQ25DLE9BQU8sSUFBSSxrQ0FBZSxDQUFDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FBQyxDQUFDO0FBQzVDLENBQUM7QUFFRCxrQkFBZSxhQUFhLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBzcmMvaW5kZXgudHMgLSDkuLvlhaXlj6NcblxuZXhwb3J0ICogZnJvbSAnLi9taWRkbGV3YXJlJztcbmV4cG9ydCAqIGZyb20gJy4vdmFsaWRhdG9ycyc7XG5leHBvcnQgKiBmcm9tICcuL2xvZ2dlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi91dGlscyc7XG5leHBvcnQgKiBmcm9tICcuL2F1ZGl0b3JzJztcblxuLy8g5L6/5o235bel5Y6C5Ye95pWwXG5pbXBvcnQgeyBMb29wRGV0ZWN0b3IgfSBmcm9tICcuL21pZGRsZXdhcmUvbG9vcC1kZXRlY3Rvcic7XG5pbXBvcnQgeyBUcmFjZUxvZ2dlciB9IGZyb20gJy4vbWlkZGxld2FyZS90cmFjZS1sb2dnZXInO1xuaW1wb3J0IHsgQ29udGV4dEluamVjdG9yIH0gZnJvbSAnLi9taWRkbGV3YXJlL2NvbnRleHQtaW5qZWN0b3InO1xuaW1wb3J0IHsgQ29uZmlnVmFsaWRhdG9yIH0gZnJvbSAnLi92YWxpZGF0b3JzL2NvbmZpZy12YWxpZGF0b3InO1xuaW1wb3J0IHsgQ29uZmlnTG9nZ2VyIH0gZnJvbSAnLi9sb2dnZXJzL2NvbmZpZy1sb2dnZXInO1xuaW1wb3J0IHsgT3BlcmF0aW9uTG9nZ2VyIH0gZnJvbSAnLi9sb2dnZXJzL29wZXJhdGlvbi1sb2dnZXInO1xuXG5leHBvcnQgaW50ZXJmYWNlIEhhcm5lc3Mge1xuICBsb29wRGV0ZWN0b3I6IExvb3BEZXRlY3RvcjtcbiAgdHJhY2VMb2dnZXI6IFRyYWNlTG9nZ2VyO1xuICBjb250ZXh0SW5qZWN0b3I6IENvbnRleHRJbmplY3RvcjtcbiAgY29uZmlnVmFsaWRhdG9yOiBDb25maWdWYWxpZGF0b3I7XG4gIGNvbmZpZ0xvZ2dlcjogQ29uZmlnTG9nZ2VyO1xuICBvcGVyYXRpb25Mb2dnZXI6IE9wZXJhdGlvbkxvZ2dlcjtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUhhcm5lc3Mod29ya3NwYWNlUGF0aDogc3RyaW5nKTogSGFybmVzcyB7XG4gIHJldHVybiB7XG4gICAgbG9vcERldGVjdG9yOiBuZXcgTG9vcERldGVjdG9yKHdvcmtzcGFjZVBhdGgpLFxuICAgIHRyYWNlTG9nZ2VyOiBuZXcgVHJhY2VMb2dnZXIod29ya3NwYWNlUGF0aCksXG4gICAgY29udGV4dEluamVjdG9yOiBuZXcgQ29udGV4dEluamVjdG9yKHdvcmtzcGFjZVBhdGgpLFxuICAgIGNvbmZpZ1ZhbGlkYXRvcjogbmV3IENvbmZpZ1ZhbGlkYXRvcih3b3Jrc3BhY2VQYXRoKSxcbiAgICBjb25maWdMb2dnZXI6IG5ldyBDb25maWdMb2dnZXIod29ya3NwYWNlUGF0aCksXG4gICAgb3BlcmF0aW9uTG9nZ2VyOiBuZXcgT3BlcmF0aW9uTG9nZ2VyKHdvcmtzcGFjZVBhdGgpLFxuICB9O1xufVxuXG4vLyDkvr/mjbflr7zlh7rvvIjnlKjkuo7ljIXoo4XlsYLvvIlcbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVDb25maWdWYWxpZGF0b3IoKTogQ29uZmlnVmFsaWRhdG9yIHtcbiAgcmV0dXJuIG5ldyBDb25maWdWYWxpZGF0b3IocHJvY2Vzcy5jd2QoKSk7XG59XG5cbmV4cG9ydCBkZWZhdWx0IGNyZWF0ZUhhcm5lc3M7XG4iXX0=
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
export interface ConfigChange {
|
|
2
|
+
timestamp: string;
|
|
3
|
+
modifier: string;
|
|
4
|
+
file: string;
|
|
5
|
+
changes: Array<{
|
|
6
|
+
field: string;
|
|
7
|
+
oldValue: any;
|
|
8
|
+
newValue: any;
|
|
9
|
+
}>;
|
|
10
|
+
reason: string;
|
|
11
|
+
verified: boolean;
|
|
12
|
+
gatewayRestarted: boolean;
|
|
13
|
+
sessionId?: string;
|
|
14
|
+
}
|
|
15
|
+
export declare class ConfigLogger {
|
|
16
|
+
private workspacePath;
|
|
17
|
+
private logFilePath;
|
|
18
|
+
constructor(workspacePath: string);
|
|
19
|
+
log(change: Omit<ConfigChange, 'timestamp'>): void;
|
|
20
|
+
private formatEntry;
|
|
21
|
+
private truncateValue;
|
|
22
|
+
getRecentChanges(limit?: number): ConfigChange[];
|
|
23
|
+
private parseEntry;
|
|
24
|
+
exportToJson(outputPath: string): void;
|
|
25
|
+
}
|