@hddz/plugin-harness 0.1.19 → 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 -10
- package/openclaw.plugin.json +0 -39
package/README.md
CHANGED
|
@@ -1,45 +1,46 @@
|
|
|
1
1
|
# @hddz/plugin-harness
|
|
2
2
|
|
|
3
|
-
>
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
> **许可**: MIT
|
|
3
|
+
> OpenClaw Harness Engineering 插件 - 为 AI Agent 构建约束、反馈与控制系统
|
|
4
|
+
|
|
5
|
+
**最新版本**: v0.2.0 (2026-03-24)
|
|
7
6
|
|
|
8
7
|
---
|
|
9
8
|
|
|
10
|
-
## 🎯
|
|
9
|
+
## 🎯 功能
|
|
11
10
|
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
| **Trace 追踪** | 完整的工具调用链路追踪 | ✅ |
|
|
11
|
+
- ✅ **配置验证**:修改 `openclaw.json` 前自动验证
|
|
12
|
+
- ✅ **Skill 审核**:安装 Skill 前自动安全扫描
|
|
13
|
+
- ✅ **循环检测**:防止 AI 反复修改同一文件
|
|
14
|
+
- ✅ **文件保护**:保护关键文件(SOUL.md 等)不被误删
|
|
15
|
+
- ✅ **操作审计**:记录所有关键操作
|
|
16
|
+
- ✅ **Trace 追踪**:完整的工具调用链路
|
|
17
|
+
- ✅ **文件监听器** (v0.2.0 新增):实时监听配置文件变化,用户手动编辑也能触发验证
|
|
20
18
|
|
|
21
19
|
---
|
|
22
20
|
|
|
23
21
|
## 🚀 快速开始
|
|
24
22
|
|
|
25
|
-
###
|
|
23
|
+
### 安装
|
|
26
24
|
|
|
27
25
|
```bash
|
|
28
|
-
#
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
# 重启 Gateway
|
|
32
|
-
openclaw gateway restart
|
|
26
|
+
# 从 npm 安装(推荐)
|
|
27
|
+
npm install @hddz/plugin-harness@0.2.0
|
|
33
28
|
|
|
34
|
-
#
|
|
35
|
-
openclaw
|
|
29
|
+
# 本地开发版
|
|
30
|
+
cd /Users/hzl/.openclaw/extensions/harness
|
|
31
|
+
npm link
|
|
36
32
|
```
|
|
37
33
|
|
|
38
|
-
|
|
34
|
+
### 配置
|
|
35
|
+
|
|
36
|
+
编辑 `~/.openclaw/openclaw.json`:
|
|
39
37
|
|
|
40
38
|
```json
|
|
41
39
|
{
|
|
42
40
|
"plugins": {
|
|
41
|
+
"installs": [
|
|
42
|
+
"@hddz/plugin-harness@0.2.0"
|
|
43
|
+
],
|
|
43
44
|
"entries": {
|
|
44
45
|
"harness": {
|
|
45
46
|
"enabled": true,
|
|
@@ -55,7 +56,7 @@ openclaw plugins list
|
|
|
55
56
|
"MEMORY.md",
|
|
56
57
|
"openclaw.json"
|
|
57
58
|
],
|
|
58
|
-
"logsDir": "
|
|
59
|
+
"logsDir": "logs/harness"
|
|
59
60
|
}
|
|
60
61
|
}
|
|
61
62
|
}
|
|
@@ -63,13 +64,9 @@ openclaw plugins list
|
|
|
63
64
|
}
|
|
64
65
|
```
|
|
65
66
|
|
|
66
|
-
|
|
67
|
+
然后重启 gateway:
|
|
67
68
|
|
|
68
69
|
```bash
|
|
69
|
-
# 链接本地插件目录
|
|
70
|
-
openclaw plugins install -l /Users/hzl/.openclaw/workspace/harness-plugin
|
|
71
|
-
|
|
72
|
-
# 重启 Gateway
|
|
73
70
|
openclaw gateway restart
|
|
74
71
|
```
|
|
75
72
|
|
|
@@ -79,17 +76,35 @@ openclaw gateway restart
|
|
|
79
76
|
|
|
80
77
|
| 选项 | 类型 | 默认值 | 说明 |
|
|
81
78
|
|------|------|--------|------|
|
|
82
|
-
| `autoValidateConfig` | boolean |
|
|
83
|
-
| `autoAuditSkill` | boolean |
|
|
84
|
-
| `loopDetectionEnabled` | boolean |
|
|
85
|
-
| `protectedFiles` | string[] |
|
|
86
|
-
| `logsDir` | string |
|
|
79
|
+
| `autoValidateConfig` | boolean | true | 配置修改前自动验证 |
|
|
80
|
+
| `autoAuditSkill` | boolean | true | Skill 安装前自动审核 |
|
|
81
|
+
| `loopDetectionEnabled` | boolean | true | 启用循环检测 |
|
|
82
|
+
| `protectedFiles` | string[] | 见上文 | 受保护的文件列表 |
|
|
83
|
+
| `logsDir` | string | logs/harness | 日志目录 |
|
|
87
84
|
|
|
88
85
|
---
|
|
89
86
|
|
|
90
|
-
##
|
|
87
|
+
## 🔍 使用示例
|
|
88
|
+
|
|
89
|
+
### 配置验证(自动)
|
|
90
|
+
|
|
91
|
+
```typescript
|
|
92
|
+
// 修改 openclaw.json 时自动触发
|
|
93
|
+
const result = await plugin.onConfigChange(newConfig);
|
|
94
|
+
// 如果验证失败,会抛出异常阻止修改
|
|
95
|
+
```
|
|
96
|
+
|
|
97
|
+
### Skill 审核(自动)
|
|
98
|
+
|
|
99
|
+
```typescript
|
|
100
|
+
// 安装 Skill 时自动触发
|
|
101
|
+
const auditResult = await plugin.onSkillInstall('/path/to/skill');
|
|
102
|
+
if (!auditResult.passed) {
|
|
103
|
+
throw new Error(`Skill 审核未通过:${auditResult.riskLevel}`);
|
|
104
|
+
}
|
|
105
|
+
```
|
|
91
106
|
|
|
92
|
-
|
|
107
|
+
### 手动查询
|
|
93
108
|
|
|
94
109
|
```bash
|
|
95
110
|
# 查看最近的配置变更
|
|
@@ -97,112 +112,80 @@ harness get-config-changes -l 10
|
|
|
97
112
|
|
|
98
113
|
# 查看操作日志
|
|
99
114
|
harness get-operations -l 20
|
|
115
|
+
|
|
116
|
+
# 查看循环检测统计
|
|
117
|
+
harness loop-stats
|
|
100
118
|
```
|
|
101
119
|
|
|
102
120
|
---
|
|
103
121
|
|
|
104
|
-
##
|
|
122
|
+
## 🛡️ 安全机制
|
|
105
123
|
|
|
106
|
-
###
|
|
124
|
+
### 保护文件
|
|
107
125
|
|
|
108
|
-
|
|
109
|
-
|------|----------|------|
|
|
110
|
-
| `onConfigChange` | 修改 `openclaw.json` 前 | 验证配置有效性 |
|
|
111
|
-
| `onSkillInstall` | 安装 Skill 前 | 扫描安全风险 |
|
|
112
|
-
| `onFileEdit` | 编辑文件时 | 循环检测 + 保护文件监控 |
|
|
113
|
-
| `onFileDelete` | 删除文件前 | 拦截保护文件删除 |
|
|
114
|
-
| `onExecCommand` | 执行命令前 | 记录审计日志 |
|
|
126
|
+
以下文件默认受保护,修改/删除前会告警:
|
|
115
127
|
|
|
116
|
-
|
|
128
|
+
- `SOUL.md` - 身份定义
|
|
129
|
+
- `USER.md` - 用户信息
|
|
130
|
+
- `AGENTS.md` - Agent 指南
|
|
131
|
+
- `TOOLS.md` - 工具配置
|
|
132
|
+
- `MEMORY.md` - 长期记忆
|
|
133
|
+
- `openclaw.json` - 核心配置
|
|
117
134
|
|
|
118
|
-
|
|
119
|
-
harness-plugin/
|
|
120
|
-
├── src/
|
|
121
|
-
│ └── index.ts # 插件核心逻辑
|
|
122
|
-
├── dist/ # 编译输出
|
|
123
|
-
├── docs/ # 文档
|
|
124
|
-
├── openclaw.plugin.json # OpenClaw 插件元数据
|
|
125
|
-
├── package.json # NPM 配置
|
|
126
|
-
├── tsconfig.json # TypeScript 配置
|
|
127
|
-
└── README.md # 本文件
|
|
128
|
-
```
|
|
135
|
+
### 风险等级
|
|
129
136
|
|
|
130
|
-
|
|
137
|
+
| 等级 | 标识 | 处理 |
|
|
138
|
+
|------|------|------|
|
|
139
|
+
| LOW | 🟢 | 允许 |
|
|
140
|
+
| MEDIUM | 🟡 | 警告,但允许 |
|
|
141
|
+
| HIGH | 🟠 | 阻止,需人工确认 |
|
|
142
|
+
| CRITICAL | 🔴 | 阻止,禁止操作 |
|
|
131
143
|
|
|
132
|
-
|
|
144
|
+
---
|
|
133
145
|
|
|
134
|
-
|
|
146
|
+
## 📚 文档
|
|
135
147
|
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
3. 确认 `openclaw.json` 配置了 `plugins.entries.harness`
|
|
140
|
-
4. 检查 manifest:`cat ~/.openclaw/extensions/harness/openclaw.plugin.json`
|
|
148
|
+
- [使用指南](./docs/使用指南.md)
|
|
149
|
+
- [开发文档](./docs/开发文档.md)
|
|
150
|
+
- [API 参考](./docs/API.md)
|
|
141
151
|
|
|
142
|
-
|
|
152
|
+
---
|
|
143
153
|
|
|
144
|
-
|
|
145
|
-
```json
|
|
146
|
-
{
|
|
147
|
-
"plugins": {
|
|
148
|
-
"entries": {
|
|
149
|
-
"harness": {
|
|
150
|
-
"enabled": true,
|
|
151
|
-
"config": {
|
|
152
|
-
"autoValidateConfig": true
|
|
153
|
-
}
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
}
|
|
157
|
-
}
|
|
158
|
-
```
|
|
154
|
+
## 🔧 开发
|
|
159
155
|
|
|
160
|
-
|
|
156
|
+
```bash
|
|
157
|
+
# 安装依赖
|
|
158
|
+
npm install
|
|
161
159
|
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
{
|
|
165
|
-
"plugins": {
|
|
166
|
-
"entries": {
|
|
167
|
-
"harness": {
|
|
168
|
-
"config": {
|
|
169
|
-
"autoValidateConfig": false,
|
|
170
|
-
"loopDetectionEnabled": false
|
|
171
|
-
}
|
|
172
|
-
}
|
|
173
|
-
}
|
|
174
|
-
}
|
|
175
|
-
}
|
|
176
|
-
```
|
|
160
|
+
# 构建
|
|
161
|
+
npm run build
|
|
177
162
|
|
|
178
|
-
|
|
163
|
+
# 测试
|
|
164
|
+
npm test
|
|
179
165
|
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
openclaw plugins uninstall harness
|
|
183
|
-
openclaw gateway restart
|
|
166
|
+
# 链接到 OpenClaw
|
|
167
|
+
npm link
|
|
184
168
|
```
|
|
185
169
|
|
|
186
170
|
---
|
|
187
171
|
|
|
188
|
-
##
|
|
172
|
+
## 📝 变更日志
|
|
173
|
+
|
|
174
|
+
### v0.1.0 (2026-03-19)
|
|
189
175
|
|
|
190
|
-
-
|
|
191
|
-
-
|
|
176
|
+
- ✅ 初始版本
|
|
177
|
+
- ✅ 配置验证
|
|
178
|
+
- ✅ Skill 审核
|
|
179
|
+
- ✅ 循环检测
|
|
180
|
+
- ✅ 操作日志
|
|
181
|
+
- ✅ Trace 追踪
|
|
192
182
|
|
|
193
183
|
---
|
|
194
184
|
|
|
195
|
-
##
|
|
185
|
+
## 📄 许可证
|
|
196
186
|
|
|
197
|
-
|
|
198
|
-
|------|------|------|
|
|
199
|
-
| 0.1.10 | 2026-03-24 | 添加 `openclaw.plugin.json` manifest,支持 `openclaw plugins install` |
|
|
200
|
-
| 0.1.9 | 2026-03-24 | 提供 `install.sh` 一键安装脚本 |
|
|
201
|
-
| 0.1.8 | 2026-03-24 | 修复 Skill 审核路径问题 |
|
|
202
|
-
| 0.1.7 | 2026-03-23 | 初始发布到 npm |
|
|
187
|
+
MIT
|
|
203
188
|
|
|
204
189
|
---
|
|
205
190
|
|
|
206
|
-
|
|
207
|
-
**仓库**: https://github.com/hddz/plugin-harness
|
|
208
|
-
**npm**: https://www.npmjs.com/package/@hddz/plugin-harness
|
|
191
|
+
_@openclaw/plugin-harness | 蓝山 | 2026-03-19_
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import type { HarnessPlugin } from './index.js';
|
|
2
|
+
export interface FileWatcherConfig {
|
|
3
|
+
watchFiles: string[];
|
|
4
|
+
debounceMs: number;
|
|
5
|
+
onFileChange: (filePath: string, newContent: any) => Promise<void>;
|
|
6
|
+
}
|
|
7
|
+
export declare class FileWatcher {
|
|
8
|
+
private watchPaths;
|
|
9
|
+
private debounceTimers;
|
|
10
|
+
private config;
|
|
11
|
+
private plugin;
|
|
12
|
+
constructor(plugin: HarnessPlugin, config: FileWatcherConfig);
|
|
13
|
+
/**
|
|
14
|
+
* 开始监听文件
|
|
15
|
+
*/
|
|
16
|
+
start(): void;
|
|
17
|
+
/**
|
|
18
|
+
* 监听单个文件
|
|
19
|
+
*/
|
|
20
|
+
private watchFile;
|
|
21
|
+
/**
|
|
22
|
+
* 停止监听单个文件
|
|
23
|
+
*/
|
|
24
|
+
private unwatchFile;
|
|
25
|
+
/**
|
|
26
|
+
* 文件变化处理(带防抖)
|
|
27
|
+
*/
|
|
28
|
+
private onFileChange;
|
|
29
|
+
/**
|
|
30
|
+
* 处理文件变化
|
|
31
|
+
*/
|
|
32
|
+
private handleFileChange;
|
|
33
|
+
/**
|
|
34
|
+
* 停止所有监听
|
|
35
|
+
*/
|
|
36
|
+
stop(): void;
|
|
37
|
+
}
|
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// src/file-watcher.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.FileWatcher = void 0;
|
|
38
|
+
const fs = __importStar(require("fs"));
|
|
39
|
+
class FileWatcher {
|
|
40
|
+
watchPaths = new Map();
|
|
41
|
+
debounceTimers = new Map();
|
|
42
|
+
config;
|
|
43
|
+
plugin;
|
|
44
|
+
constructor(plugin, config) {
|
|
45
|
+
this.plugin = plugin;
|
|
46
|
+
this.config = config;
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* 开始监听文件
|
|
50
|
+
*/
|
|
51
|
+
start() {
|
|
52
|
+
console.log('👁️ [FileWatcher] 开始监听配置文件...');
|
|
53
|
+
for (const file of this.config.watchFiles) {
|
|
54
|
+
this.watchFile(file);
|
|
55
|
+
}
|
|
56
|
+
console.log(`👁️ [FileWatcher] 正在监听 ${this.config.watchFiles.length} 个文件`);
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* 监听单个文件
|
|
60
|
+
*/
|
|
61
|
+
watchFile(filePath) {
|
|
62
|
+
// 如果已经在监听,先停止
|
|
63
|
+
if (this.watchPaths.has(filePath)) {
|
|
64
|
+
this.unwatchFile(filePath);
|
|
65
|
+
}
|
|
66
|
+
try {
|
|
67
|
+
const watcher = fs.watch(filePath, { persistent: false }, (eventType, filename) => {
|
|
68
|
+
if (eventType === 'change' || eventType === 'rename') {
|
|
69
|
+
console.log(`📝 [FileWatcher] 检测到文件变化:${filePath} (${eventType})`);
|
|
70
|
+
this.onFileChange(filePath);
|
|
71
|
+
}
|
|
72
|
+
});
|
|
73
|
+
watcher.on('error', (error) => {
|
|
74
|
+
console.error(`❌ [FileWatcher] 监听错误 ${filePath}:`, error.message);
|
|
75
|
+
});
|
|
76
|
+
this.watchPaths.set(filePath, watcher);
|
|
77
|
+
console.log(`✅ [FileWatcher] 开始监听:${filePath}`);
|
|
78
|
+
}
|
|
79
|
+
catch (error) {
|
|
80
|
+
console.warn(`⚠️ [FileWatcher] 无法监听文件 ${filePath}: ${error.message}`);
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* 停止监听单个文件
|
|
85
|
+
*/
|
|
86
|
+
unwatchFile(filePath) {
|
|
87
|
+
const watcher = this.watchPaths.get(filePath);
|
|
88
|
+
if (watcher) {
|
|
89
|
+
watcher.close();
|
|
90
|
+
this.watchPaths.delete(filePath);
|
|
91
|
+
console.log(`🛑 [FileWatcher] 停止监听:${filePath}`);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* 文件变化处理(带防抖)
|
|
96
|
+
*/
|
|
97
|
+
async onFileChange(filePath) {
|
|
98
|
+
// 清除之前的定时器
|
|
99
|
+
const existingTimer = this.debounceTimers.get(filePath);
|
|
100
|
+
if (existingTimer) {
|
|
101
|
+
clearTimeout(existingTimer);
|
|
102
|
+
}
|
|
103
|
+
// 设置新的防抖定时器
|
|
104
|
+
const timer = setTimeout(async () => {
|
|
105
|
+
try {
|
|
106
|
+
await this.handleFileChange(filePath);
|
|
107
|
+
}
|
|
108
|
+
catch (error) {
|
|
109
|
+
console.error(`❌ [FileWatcher] 处理文件变化失败 ${filePath}:`, error.message);
|
|
110
|
+
}
|
|
111
|
+
this.debounceTimers.delete(filePath);
|
|
112
|
+
}, this.config.debounceMs);
|
|
113
|
+
this.debounceTimers.set(filePath, timer);
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* 处理文件变化
|
|
117
|
+
*/
|
|
118
|
+
async handleFileChange(filePath) {
|
|
119
|
+
console.log(`🔍 [FileWatcher] 处理文件变化:${filePath}`);
|
|
120
|
+
// 读取新配置
|
|
121
|
+
let newConfig;
|
|
122
|
+
try {
|
|
123
|
+
const content = fs.readFileSync(filePath, 'utf8');
|
|
124
|
+
newConfig = JSON.parse(content);
|
|
125
|
+
}
|
|
126
|
+
catch (error) {
|
|
127
|
+
console.error(`❌ [FileWatcher] 读取配置失败:${error.message}`);
|
|
128
|
+
return;
|
|
129
|
+
}
|
|
130
|
+
// 调用插件的配置变更处理
|
|
131
|
+
await this.config.onFileChange(filePath, newConfig);
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* 停止所有监听
|
|
135
|
+
*/
|
|
136
|
+
stop() {
|
|
137
|
+
console.log('🛑 [FileWatcher] 停止所有文件监听...');
|
|
138
|
+
for (const [filePath, watcher] of this.watchPaths.entries()) {
|
|
139
|
+
watcher.close();
|
|
140
|
+
}
|
|
141
|
+
this.watchPaths.clear();
|
|
142
|
+
// 清除所有定时器
|
|
143
|
+
for (const [filePath, timer] of this.debounceTimers.entries()) {
|
|
144
|
+
clearTimeout(timer);
|
|
145
|
+
}
|
|
146
|
+
this.debounceTimers.clear();
|
|
147
|
+
console.log('🛑 [FileWatcher] 已停止所有监听');
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
exports.FileWatcher = FileWatcher;
|
|
151
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-watcher.js","sourceRoot":"","sources":["../src/file-watcher.ts"],"names":[],"mappings":"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","sourcesContent":["// src/file-watcher.ts - 配置文件监听器\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport type { HarnessPlugin } from './index.js';\n\nexport interface FileWatcherConfig {\n  watchFiles: string[];\n  debounceMs: number;\n  onFileChange: (filePath: string, newContent: any) => Promise<void>;\n}\n\nexport class FileWatcher {\n  private watchPaths: Map<string, fs.FSWatcher> = new Map();\n  private debounceTimers: Map<string, NodeJS.Timeout> = new Map();\n  private config: FileWatcherConfig;\n  private plugin: HarnessPlugin;\n\n  constructor(plugin: HarnessPlugin, config: FileWatcherConfig) {\n    this.plugin = plugin;\n    this.config = config;\n  }\n\n  /**\n   * 开始监听文件\n   */\n  start(): void {\n    console.log('👁️ [FileWatcher] 开始监听配置文件...');\n    \n    for (const file of this.config.watchFiles) {\n      this.watchFile(file);\n    }\n    \n    console.log(`👁️ [FileWatcher] 正在监听 ${this.config.watchFiles.length} 个文件`);\n  }\n\n  /**\n   * 监听单个文件\n   */\n  private watchFile(filePath: string): void {\n    // 如果已经在监听，先停止\n    if (this.watchPaths.has(filePath)) {\n      this.unwatchFile(filePath);\n    }\n\n    try {\n      const watcher = fs.watch(filePath, { persistent: false }, (eventType, filename) => {\n        if (eventType === 'change' || eventType === 'rename') {\n          console.log(`📝 [FileWatcher] 检测到文件变化：${filePath} (${eventType})`);\n          this.onFileChange(filePath);\n        }\n      });\n\n      watcher.on('error', (error) => {\n        console.error(`❌ [FileWatcher] 监听错误 ${filePath}:`, error.message);\n      });\n\n      this.watchPaths.set(filePath, watcher);\n      console.log(`✅ [FileWatcher] 开始监听：${filePath}`);\n    } catch (error: any) {\n      console.warn(`⚠️ [FileWatcher] 无法监听文件 ${filePath}: ${error.message}`);\n    }\n  }\n\n  /**\n   * 停止监听单个文件\n   */\n  private unwatchFile(filePath: string): void {\n    const watcher = this.watchPaths.get(filePath);\n    if (watcher) {\n      watcher.close();\n      this.watchPaths.delete(filePath);\n      console.log(`🛑 [FileWatcher] 停止监听：${filePath}`);\n    }\n  }\n\n  /**\n   * 文件变化处理（带防抖）\n   */\n  private async onFileChange(filePath: string): Promise<void> {\n    // 清除之前的定时器\n    const existingTimer = this.debounceTimers.get(filePath);\n    if (existingTimer) {\n      clearTimeout(existingTimer);\n    }\n\n    // 设置新的防抖定时器\n    const timer = setTimeout(async () => {\n      try {\n        await this.handleFileChange(filePath);\n      } catch (error: any) {\n        console.error(`❌ [FileWatcher] 处理文件变化失败 ${filePath}:`, error.message);\n      }\n      this.debounceTimers.delete(filePath);\n    }, this.config.debounceMs);\n\n    this.debounceTimers.set(filePath, timer);\n  }\n\n  /**\n   * 处理文件变化\n   */\n  private async handleFileChange(filePath: string): Promise<void> {\n    console.log(`🔍 [FileWatcher] 处理文件变化：${filePath}`);\n\n    // 读取新配置\n    let newConfig: any;\n    try {\n      const content = fs.readFileSync(filePath, 'utf8');\n      newConfig = JSON.parse(content);\n    } catch (error: any) {\n      console.error(`❌ [FileWatcher] 读取配置失败：${error.message}`);\n      return;\n    }\n\n    // 调用插件的配置变更处理\n    await this.config.onFileChange(filePath, newConfig);\n  }\n\n  /**\n   * 停止所有监听\n   */\n  stop(): void {\n    console.log('🛑 [FileWatcher] 停止所有文件监听...');\n    \n    for (const [filePath, watcher] of this.watchPaths.entries()) {\n      watcher.close();\n    }\n    \n    this.watchPaths.clear();\n    \n    // 清除所有定时器\n    for (const [filePath, timer] of this.debounceTimers.entries()) {\n      clearTimeout(timer);\n    }\n    \n    this.debounceTimers.clear();\n    console.log('🛑 [FileWatcher] 已停止所有监听');\n  }\n}\n"]}
|
package/dist/index.d.ts
CHANGED
|
@@ -5,3 +5,66 @@ export interface HarnessPluginConfig {
|
|
|
5
5
|
protectedFiles: string[];
|
|
6
6
|
logsDir: string;
|
|
7
7
|
}
|
|
8
|
+
export declare class HarnessPlugin {
|
|
9
|
+
name: string;
|
|
10
|
+
version: string;
|
|
11
|
+
private config;
|
|
12
|
+
private workspacePath;
|
|
13
|
+
private configValidator;
|
|
14
|
+
private skillAuditor;
|
|
15
|
+
private loopDetector;
|
|
16
|
+
private configLogger;
|
|
17
|
+
private operationLogger;
|
|
18
|
+
private traceLogger;
|
|
19
|
+
constructor(workspacePath: string, config?: Partial<HarnessPluginConfig>);
|
|
20
|
+
/**
|
|
21
|
+
* Hook: 配置修改前验证
|
|
22
|
+
*/
|
|
23
|
+
onConfigChange(newConfig: any, sessionId?: string): Promise<{
|
|
24
|
+
valid: boolean;
|
|
25
|
+
errors?: string[];
|
|
26
|
+
}>;
|
|
27
|
+
/**
|
|
28
|
+
* 公开方法:记录配置变更
|
|
29
|
+
*/
|
|
30
|
+
logConfigChange(entry: {
|
|
31
|
+
sessionId: string;
|
|
32
|
+
modifier: string;
|
|
33
|
+
file: string;
|
|
34
|
+
reason: string;
|
|
35
|
+
verified: boolean;
|
|
36
|
+
gatewayRestarted: boolean;
|
|
37
|
+
changes: any[];
|
|
38
|
+
}): void;
|
|
39
|
+
/**
|
|
40
|
+
* Hook: Skill 安装前审核
|
|
41
|
+
*/
|
|
42
|
+
onSkillInstall(skillPath: string, source?: string, sessionId?: string): Promise<{
|
|
43
|
+
passed: boolean;
|
|
44
|
+
warnings?: any[];
|
|
45
|
+
}>;
|
|
46
|
+
/**
|
|
47
|
+
* Hook: 文件编辑时循环检测
|
|
48
|
+
*/
|
|
49
|
+
onFileEdit(filePath: string, sessionId?: string): {
|
|
50
|
+
allowed: boolean;
|
|
51
|
+
warning?: string;
|
|
52
|
+
};
|
|
53
|
+
/**
|
|
54
|
+
* Hook: 文件删除前检查
|
|
55
|
+
*/
|
|
56
|
+
onFileDelete(filePath: string, sessionId?: string): {
|
|
57
|
+
allowed: boolean;
|
|
58
|
+
reason?: string;
|
|
59
|
+
};
|
|
60
|
+
/**
|
|
61
|
+
* Hook: 系统命令执行前
|
|
62
|
+
*/
|
|
63
|
+
onExecCommand(command: string, sessionId?: string): void;
|
|
64
|
+
/**
|
|
65
|
+
* 获取插件状态
|
|
66
|
+
*/
|
|
67
|
+
getStatus(): any;
|
|
68
|
+
}
|
|
69
|
+
export declare function createPlugin(workspacePath: string, config: any): HarnessPlugin;
|
|
70
|
+
export default HarnessPlugin;
|