@hddz/plugin-harness 0.1.18 → 0.2.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.
Potentially problematic release.
This version of @hddz/plugin-harness might be problematic. Click here for more details.
- package/README.md +97 -114
- 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 -15
- package/openclaw.plugin.json +0 -39
|
@@ -0,0 +1,192 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// src/index.ts - Harness Plugin 主入口
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.HarnessPlugin = void 0;
|
|
5
|
+
exports.createPlugin = createPlugin;
|
|
6
|
+
const harness_1 = require("@openclaw/harness");
|
|
7
|
+
const DEFAULT_CONFIG = {
|
|
8
|
+
autoValidateConfig: true,
|
|
9
|
+
autoAuditSkill: true,
|
|
10
|
+
loopDetectionEnabled: true,
|
|
11
|
+
protectedFiles: [
|
|
12
|
+
'SOUL.md',
|
|
13
|
+
'USER.md',
|
|
14
|
+
'AGENTS.md',
|
|
15
|
+
'TOOLS.md',
|
|
16
|
+
'MEMORY.md',
|
|
17
|
+
'openclaw.json',
|
|
18
|
+
],
|
|
19
|
+
logsDir: 'logs/harness', // 相对 workspace 的路径
|
|
20
|
+
};
|
|
21
|
+
class HarnessPlugin {
|
|
22
|
+
name = 'harness';
|
|
23
|
+
version = '0.1.0';
|
|
24
|
+
config;
|
|
25
|
+
workspacePath;
|
|
26
|
+
configValidator;
|
|
27
|
+
skillAuditor;
|
|
28
|
+
loopDetector;
|
|
29
|
+
configLogger;
|
|
30
|
+
operationLogger;
|
|
31
|
+
traceLogger;
|
|
32
|
+
constructor(workspacePath, config = {}) {
|
|
33
|
+
this.workspacePath = workspacePath;
|
|
34
|
+
this.config = { ...DEFAULT_CONFIG, ...config };
|
|
35
|
+
// 初始化工具
|
|
36
|
+
this.configValidator = new harness_1.ConfigValidator(workspacePath);
|
|
37
|
+
this.skillAuditor = new harness_1.SkillAuditor(workspacePath);
|
|
38
|
+
this.loopDetector = new harness_1.LoopDetector(workspacePath);
|
|
39
|
+
this.configLogger = new harness_1.ConfigLogger(workspacePath);
|
|
40
|
+
this.operationLogger = new harness_1.OperationLogger(workspacePath);
|
|
41
|
+
this.traceLogger = new harness_1.TraceLogger(workspacePath);
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Hook: 配置修改前验证
|
|
45
|
+
*/
|
|
46
|
+
async onConfigChange(newConfig, sessionId) {
|
|
47
|
+
if (!this.config.autoValidateConfig) {
|
|
48
|
+
return { valid: true };
|
|
49
|
+
}
|
|
50
|
+
console.log('🔍 [Harness] 验证配置变更...');
|
|
51
|
+
const result = this.configValidator.validateContent(newConfig);
|
|
52
|
+
if (!result.valid) {
|
|
53
|
+
console.error('❌ [Harness] 配置验证失败:');
|
|
54
|
+
result.errors.forEach(e => console.error(` - ${e}`));
|
|
55
|
+
// 记录配置变更失败
|
|
56
|
+
this.configLogger.log({
|
|
57
|
+
sessionId: sessionId || 'unknown',
|
|
58
|
+
modifier: 'AI Agent',
|
|
59
|
+
file: 'openclaw.json',
|
|
60
|
+
reason: '配置修改(验证失败)',
|
|
61
|
+
verified: false,
|
|
62
|
+
gatewayRestarted: false,
|
|
63
|
+
changes: [],
|
|
64
|
+
});
|
|
65
|
+
return { valid: false, errors: result.errors };
|
|
66
|
+
}
|
|
67
|
+
console.log('✅ [Harness] 配置验证通过');
|
|
68
|
+
// 记录配置变更成功
|
|
69
|
+
this.configLogger.log({
|
|
70
|
+
sessionId: sessionId || 'unknown',
|
|
71
|
+
modifier: 'AI Agent',
|
|
72
|
+
file: 'openclaw.json',
|
|
73
|
+
reason: '配置修改',
|
|
74
|
+
verified: true,
|
|
75
|
+
gatewayRestarted: true,
|
|
76
|
+
changes: [],
|
|
77
|
+
});
|
|
78
|
+
return { valid: true };
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* 公开方法:记录配置变更
|
|
82
|
+
*/
|
|
83
|
+
logConfigChange(entry) {
|
|
84
|
+
this.configLogger.log(entry);
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Hook: Skill 安装前审核
|
|
88
|
+
*/
|
|
89
|
+
async onSkillInstall(skillPath, source, sessionId) {
|
|
90
|
+
if (!this.config.autoAuditSkill) {
|
|
91
|
+
return { passed: true };
|
|
92
|
+
}
|
|
93
|
+
console.log(`🔍 [Harness] 审核 Skill: ${skillPath}`);
|
|
94
|
+
const result = await this.skillAuditor.audit(skillPath, source || 'unknown');
|
|
95
|
+
if (!result.passed) {
|
|
96
|
+
console.error(`❌ [Harness] Skill 审核未通过:${result.riskLevel}`);
|
|
97
|
+
result.warnings.forEach(w => console.error(` - ${w.message}`));
|
|
98
|
+
return { passed: false, warnings: result.warnings };
|
|
99
|
+
}
|
|
100
|
+
console.log(`✅ [Harness] Skill 审核通过:${result.riskLevel}`);
|
|
101
|
+
return { passed: true };
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Hook: 文件编辑时循环检测
|
|
105
|
+
*/
|
|
106
|
+
onFileEdit(filePath, sessionId) {
|
|
107
|
+
if (!this.config.loopDetectionEnabled) {
|
|
108
|
+
return { allowed: true };
|
|
109
|
+
}
|
|
110
|
+
const relativePath = filePath.split('/').pop() || filePath;
|
|
111
|
+
// 检查是否是保护文件
|
|
112
|
+
if (this.config.protectedFiles.some(f => filePath.endsWith(f))) {
|
|
113
|
+
console.warn(`⚠️ [Harness] 保护文件被修改:${filePath}`);
|
|
114
|
+
// 记录保护文件修改
|
|
115
|
+
this.operationLogger.log({
|
|
116
|
+
sessionId: sessionId || 'unknown',
|
|
117
|
+
modifier: 'AI Agent',
|
|
118
|
+
type: 'file_edit',
|
|
119
|
+
description: `保护文件修改:${filePath}`,
|
|
120
|
+
file: filePath,
|
|
121
|
+
result: 'success',
|
|
122
|
+
});
|
|
123
|
+
}
|
|
124
|
+
// 循环检测
|
|
125
|
+
const warning = this.loopDetector.onFileEdit(filePath);
|
|
126
|
+
if (warning) {
|
|
127
|
+
console.warn(`⚠️ [Harness] 循环编辑警告:${warning}`);
|
|
128
|
+
return { allowed: true, warning }; // 允许但警告
|
|
129
|
+
}
|
|
130
|
+
return { allowed: true };
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* Hook: 文件删除前检查
|
|
134
|
+
*/
|
|
135
|
+
onFileDelete(filePath, sessionId) {
|
|
136
|
+
const relativePath = filePath.split('/').pop() || filePath;
|
|
137
|
+
// 检查是否是保护文件
|
|
138
|
+
if (this.config.protectedFiles.some(f => filePath.endsWith(f))) {
|
|
139
|
+
console.error(`🚨 [Harness] 禁止删除保护文件:${filePath}`);
|
|
140
|
+
this.operationLogger.log({
|
|
141
|
+
sessionId: sessionId || 'unknown',
|
|
142
|
+
modifier: 'AI Agent',
|
|
143
|
+
type: 'file_delete',
|
|
144
|
+
description: `尝试删除保护文件:${filePath}`,
|
|
145
|
+
file: filePath,
|
|
146
|
+
result: 'error',
|
|
147
|
+
});
|
|
148
|
+
return { allowed: false, reason: '保护文件禁止删除' };
|
|
149
|
+
}
|
|
150
|
+
this.operationLogger.log({
|
|
151
|
+
sessionId: sessionId || 'unknown',
|
|
152
|
+
modifier: 'AI Agent',
|
|
153
|
+
type: 'file_delete',
|
|
154
|
+
description: `删除文件:${filePath}`,
|
|
155
|
+
file: filePath,
|
|
156
|
+
result: 'success',
|
|
157
|
+
});
|
|
158
|
+
return { allowed: true };
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Hook: 系统命令执行前
|
|
162
|
+
*/
|
|
163
|
+
onExecCommand(command, sessionId) {
|
|
164
|
+
this.operationLogger.log({
|
|
165
|
+
sessionId: sessionId || 'unknown',
|
|
166
|
+
modifier: 'AI Agent',
|
|
167
|
+
type: 'system_command',
|
|
168
|
+
description: `执行命令:${command}`,
|
|
169
|
+
command,
|
|
170
|
+
result: 'pending',
|
|
171
|
+
});
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* 获取插件状态
|
|
175
|
+
*/
|
|
176
|
+
getStatus() {
|
|
177
|
+
return {
|
|
178
|
+
name: this.name,
|
|
179
|
+
version: this.version,
|
|
180
|
+
enabled: true,
|
|
181
|
+
config: this.config,
|
|
182
|
+
loopStats: this.loopDetector.getStats(),
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
exports.HarnessPlugin = HarnessPlugin;
|
|
187
|
+
// OpenClaw 插件工厂函数
|
|
188
|
+
function createPlugin(workspacePath, config) {
|
|
189
|
+
return new HarnessPlugin(workspacePath, config);
|
|
190
|
+
}
|
|
191
|
+
exports.default = HarnessPlugin;
|
|
192
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["// src/index.ts - Harness Plugin 主入口\n\nimport { ConfigValidator, SkillAuditor, LoopDetector, ConfigLogger, OperationLogger, TraceLogger } from '@openclaw/harness';\n\nexport interface HarnessPluginConfig {\n  autoValidateConfig: boolean;\n  autoAuditSkill: boolean;\n  loopDetectionEnabled: boolean;\n  protectedFiles: string[];\n  logsDir: string;\n}\n\nconst DEFAULT_CONFIG: HarnessPluginConfig = {\n  autoValidateConfig: true,\n  autoAuditSkill: true,\n  loopDetectionEnabled: true,\n  protectedFiles: [\n    'SOUL.md',\n    'USER.md',\n    'AGENTS.md',\n    'TOOLS.md',\n    'MEMORY.md',\n    'openclaw.json',\n  ],\n  logsDir: 'logs/harness',  // 相对 workspace 的路径\n};\n\nexport class HarnessPlugin {\n  name = 'harness';\n  version = '0.1.0';\n  \n  private config: HarnessPluginConfig;\n  private workspacePath: string;\n  private configValidator: ConfigValidator;\n  private skillAuditor: SkillAuditor;\n  private loopDetector: LoopDetector;\n  private configLogger: ConfigLogger;\n  private operationLogger: OperationLogger;\n  private traceLogger: TraceLogger;\n\n  constructor(workspacePath: string, config: Partial<HarnessPluginConfig> = {}) {\n    this.workspacePath = workspacePath;\n    this.config = { ...DEFAULT_CONFIG, ...config };\n    \n    // 初始化工具\n    this.configValidator = new ConfigValidator(workspacePath);\n    this.skillAuditor = new SkillAuditor(workspacePath);\n    this.loopDetector = new LoopDetector(workspacePath);\n    this.configLogger = new ConfigLogger(workspacePath);\n    this.operationLogger = new OperationLogger(workspacePath);\n    this.traceLogger = new TraceLogger(workspacePath);\n  }\n\n  /**\n   * Hook: 配置修改前验证\n   */\n  async onConfigChange(newConfig: any, sessionId?: string): Promise<{ valid: boolean; errors?: string[] }> {\n    if (!this.config.autoValidateConfig) {\n      return { valid: true };\n    }\n\n    console.log('🔍 [Harness] 验证配置变更...');\n    \n    const result = this.configValidator.validateContent(newConfig);\n    \n    if (!result.valid) {\n      console.error('❌ [Harness] 配置验证失败:');\n      result.errors.forEach(e => console.error(`   - ${e}`));\n      \n      // 记录配置变更失败\n      this.configLogger.log({\n        sessionId: sessionId || 'unknown',\n        modifier: 'AI Agent',\n        file: 'openclaw.json',\n        reason: '配置修改（验证失败）',\n        verified: false,\n        gatewayRestarted: false,\n        changes: [],\n      });\n      \n      return { valid: false, errors: result.errors };\n    }\n    \n    console.log('✅ [Harness] 配置验证通过');\n    \n    // 记录配置变更成功\n    this.configLogger.log({\n      sessionId: sessionId || 'unknown',\n      modifier: 'AI Agent',\n      file: 'openclaw.json',\n      reason: '配置修改',\n      verified: true,\n      gatewayRestarted: true,\n      changes: [],\n    });\n    \n    return { valid: true };\n  }\n\n  /**\n   * 公开方法：记录配置变更\n   */\n  logConfigChange(entry: {\n    sessionId: string;\n    modifier: string;\n    file: string;\n    reason: string;\n    verified: boolean;\n    gatewayRestarted: boolean;\n    changes: any[];\n  }) {\n    this.configLogger.log(entry);\n  }\n\n  /**\n   * Hook: Skill 安装前审核\n   */\n  async onSkillInstall(skillPath: string, source?: string, sessionId?: string): Promise<{ passed: boolean; warnings?: any[] }> {\n    if (!this.config.autoAuditSkill) {\n      return { passed: true };\n    }\n\n    console.log(`🔍 [Harness] 审核 Skill: ${skillPath}`);\n    \n    const result = await this.skillAuditor.audit(skillPath, source || 'unknown');\n    \n    if (!result.passed) {\n      console.error(`❌ [Harness] Skill 审核未通过：${result.riskLevel}`);\n      result.warnings.forEach(w => console.error(`   - ${w.message}`));\n      \n      return { passed: false, warnings: result.warnings };\n    }\n    \n    console.log(`✅ [Harness] Skill 审核通过：${result.riskLevel}`);\n    return { passed: true };\n  }\n\n  /**\n   * Hook: 文件编辑时循环检测\n   */\n  onFileEdit(filePath: string, sessionId?: string): { allowed: boolean; warning?: string } {\n    if (!this.config.loopDetectionEnabled) {\n      return { allowed: true };\n    }\n\n    const relativePath = filePath.split('/').pop() || filePath;\n    \n    // 检查是否是保护文件\n    if (this.config.protectedFiles.some(f => filePath.endsWith(f))) {\n      console.warn(`⚠️ [Harness] 保护文件被修改：${filePath}`);\n      // 记录保护文件修改\n      this.operationLogger.log({\n        sessionId: sessionId || 'unknown',\n        modifier: 'AI Agent',\n        type: 'file_edit',\n        description: `保护文件修改：${filePath}`,\n        file: filePath,\n        result: 'success',\n      });\n    }\n    \n    // 循环检测\n    const warning = this.loopDetector.onFileEdit(filePath);\n    \n    if (warning) {\n      console.warn(`⚠️ [Harness] 循环编辑警告：${warning}`);\n      return { allowed: true, warning }; // 允许但警告\n    }\n    \n    return { allowed: true };\n  }\n\n  /**\n   * Hook: 文件删除前检查\n   */\n  onFileDelete(filePath: string, sessionId?: string): { allowed: boolean; reason?: string } {\n    const relativePath = filePath.split('/').pop() || filePath;\n    \n    // 检查是否是保护文件\n    if (this.config.protectedFiles.some(f => filePath.endsWith(f))) {\n      console.error(`🚨 [Harness] 禁止删除保护文件：${filePath}`);\n      \n      this.operationLogger.log({\n        sessionId: sessionId || 'unknown',\n        modifier: 'AI Agent',\n        type: 'file_delete',\n        description: `尝试删除保护文件：${filePath}`,\n        file: filePath,\n        result: 'error',\n      });\n      \n      return { allowed: false, reason: '保护文件禁止删除' };\n    }\n    \n    this.operationLogger.log({\n      sessionId: sessionId || 'unknown',\n      modifier: 'AI Agent',\n      type: 'file_delete',\n      description: `删除文件：${filePath}`,\n      file: filePath,\n      result: 'success',\n    });\n    \n    return { allowed: true };\n  }\n\n  /**\n   * Hook: 系统命令执行前\n   */\n  onExecCommand(command: string, sessionId?: string): void {\n    this.operationLogger.log({\n      sessionId: sessionId || 'unknown',\n      modifier: 'AI Agent',\n      type: 'system_command',\n      description: `执行命令：${command}`,\n      command,\n      result: 'pending',\n    });\n  }\n\n  /**\n   * 获取插件状态\n   */\n  getStatus(): any {\n    return {\n      name: this.name,\n      version: this.version,\n      enabled: true,\n      config: this.config,\n      loopStats: this.loopDetector.getStats(),\n    };\n  }\n}\n\n// OpenClaw 插件工厂函数\nexport function createPlugin(workspacePath: string, config: any): HarnessPlugin {\n  return new HarnessPlugin(workspacePath, config);\n}\n\nexport default HarnessPlugin;\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@hddz/plugin-harness",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.2.0",
|
|
4
4
|
"description": "Harness Engineering plugin for OpenClaw - constraints, feedback loops, and control systems for AI agents",
|
|
5
5
|
"main": "dist/index.js",
|
|
6
6
|
"types": "dist/index.d.ts",
|
|
@@ -21,31 +21,20 @@
|
|
|
21
21
|
"author": "蓝山",
|
|
22
22
|
"license": "MIT",
|
|
23
23
|
"dependencies": {
|
|
24
|
-
"@hddz/harness": "^0.1
|
|
24
|
+
"@hddz/harness": "^1.0.1"
|
|
25
25
|
},
|
|
26
26
|
"devDependencies": {
|
|
27
27
|
"@types/node": "^20.0.0",
|
|
28
|
-
"typescript": "^5.0.0"
|
|
29
|
-
"jest": "^29.0.0",
|
|
30
|
-
"eslint": "^8.0.0"
|
|
28
|
+
"typescript": "^5.0.0"
|
|
31
29
|
},
|
|
32
30
|
"engines": {
|
|
33
31
|
"node": ">=18.0.0"
|
|
34
32
|
},
|
|
35
33
|
"files": [
|
|
36
34
|
"dist",
|
|
37
|
-
"README.md"
|
|
38
|
-
"openclaw.plugin.json"
|
|
35
|
+
"README.md"
|
|
39
36
|
],
|
|
40
37
|
"peerDependencies": {
|
|
41
38
|
"openclaw": ">=2026.3.0"
|
|
42
|
-
},
|
|
43
|
-
"openclaw": {
|
|
44
|
-
"extensions": [
|
|
45
|
-
{
|
|
46
|
-
"id": "harness",
|
|
47
|
-
"path": "./dist/index.js"
|
|
48
|
-
}
|
|
49
|
-
]
|
|
50
39
|
}
|
|
51
40
|
}
|
package/openclaw.plugin.json
DELETED
|
@@ -1,39 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"id": "harness",
|
|
3
|
-
"name": "Harness Engineering",
|
|
4
|
-
"description": "Constraints, feedback loops, and control systems for AI agents",
|
|
5
|
-
"version": "0.1.18",
|
|
6
|
-
"kind": "engineering",
|
|
7
|
-
"configSchema": {
|
|
8
|
-
"type": "object",
|
|
9
|
-
"additionalProperties": false,
|
|
10
|
-
"properties": {
|
|
11
|
-
"autoValidateConfig": {
|
|
12
|
-
"type": "boolean",
|
|
13
|
-
"default": true,
|
|
14
|
-
"description": "Automatically validate openclaw.json before writes"
|
|
15
|
-
},
|
|
16
|
-
"autoAuditSkill": {
|
|
17
|
-
"type": "boolean",
|
|
18
|
-
"default": true,
|
|
19
|
-
"description": "Automatically audit skills before installation"
|
|
20
|
-
},
|
|
21
|
-
"loopDetectionEnabled": {
|
|
22
|
-
"type": "boolean",
|
|
23
|
-
"default": true,
|
|
24
|
-
"description": "Detect and prevent infinite edit loops"
|
|
25
|
-
},
|
|
26
|
-
"protectedFiles": {
|
|
27
|
-
"type": "array",
|
|
28
|
-
"items": { "type": "string" },
|
|
29
|
-
"default": ["SOUL.md", "USER.md", "AGENTS.md", "TOOLS.md", "MEMORY.md", "openclaw.json"],
|
|
30
|
-
"description": "Files protected from deletion/modification"
|
|
31
|
-
},
|
|
32
|
-
"logsDir": {
|
|
33
|
-
"type": "string",
|
|
34
|
-
"default": "workspace/logs/harness",
|
|
35
|
-
"description": "Directory for harness operation logs"
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
}
|
|
39
|
-
}
|