@hddz/plugin-harness 0.2.0 → 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/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/index.js +8 -8
- package/package.json +2 -4
|
@@ -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
|
+
}
|
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// src/loggers/config-logger.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.ConfigLogger = void 0;
|
|
38
|
+
const fs = __importStar(require("fs"));
|
|
39
|
+
const path = __importStar(require("path"));
|
|
40
|
+
class ConfigLogger {
|
|
41
|
+
workspacePath;
|
|
42
|
+
logFilePath;
|
|
43
|
+
constructor(workspacePath) {
|
|
44
|
+
this.workspacePath = workspacePath;
|
|
45
|
+
const logsDir = path.join(workspacePath, 'logs');
|
|
46
|
+
if (!fs.existsSync(logsDir)) {
|
|
47
|
+
fs.mkdirSync(logsDir, { recursive: true });
|
|
48
|
+
}
|
|
49
|
+
this.logFilePath = path.join(logsDir, 'config-changes.md');
|
|
50
|
+
// 初始化日志文件(如果不存在)
|
|
51
|
+
if (!fs.existsSync(this.logFilePath)) {
|
|
52
|
+
fs.writeFileSync(this.logFilePath, '# 配置变更日志\n\n');
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
log(change) {
|
|
56
|
+
const fullChange = {
|
|
57
|
+
...change,
|
|
58
|
+
timestamp: new Date().toISOString(),
|
|
59
|
+
};
|
|
60
|
+
const entry = this.formatEntry(fullChange);
|
|
61
|
+
fs.appendFileSync(this.logFilePath, entry);
|
|
62
|
+
}
|
|
63
|
+
formatEntry(change) {
|
|
64
|
+
const date = new Date(change.timestamp);
|
|
65
|
+
const dateStr = date.toISOString().replace('T', ' ').substring(0, 19);
|
|
66
|
+
let entry = `## ${dateStr}\n`;
|
|
67
|
+
entry += `- **修改者**: ${change.modifier}\n`;
|
|
68
|
+
entry += `- **文件**: ${change.file}\n`;
|
|
69
|
+
entry += `- **原因**: ${change.reason}\n`;
|
|
70
|
+
entry += `- **验证**: ${change.verified ? '✅ 通过' : '❌ 失败'}\n`;
|
|
71
|
+
entry += `- **Gateway 重启**: ${change.gatewayRestarted ? '✅ 是' : '❌ 否'}\n`;
|
|
72
|
+
if (change.sessionId) {
|
|
73
|
+
entry += `- **会话 ID**: ${change.sessionId}\n`;
|
|
74
|
+
}
|
|
75
|
+
entry += `\n### 变更详情\n`;
|
|
76
|
+
if (change.changes.length === 0) {
|
|
77
|
+
entry += '_无具体变更记录_\n';
|
|
78
|
+
}
|
|
79
|
+
else {
|
|
80
|
+
change.changes.forEach((c, i) => {
|
|
81
|
+
entry += `${i + 1}. \`${c.field}\`\n`;
|
|
82
|
+
entry += ` - 旧值:\`${this.truncateValue(c.oldValue)}\`\n`;
|
|
83
|
+
entry += ` - 新值:\`${this.truncateValue(c.newValue)}\`\n`;
|
|
84
|
+
});
|
|
85
|
+
}
|
|
86
|
+
entry += `\n---\n\n`;
|
|
87
|
+
return entry;
|
|
88
|
+
}
|
|
89
|
+
truncateValue(value, maxLength = 100) {
|
|
90
|
+
const str = JSON.stringify(value);
|
|
91
|
+
if (str.length <= maxLength)
|
|
92
|
+
return str;
|
|
93
|
+
return str.substring(0, maxLength) + '...';
|
|
94
|
+
}
|
|
95
|
+
getRecentChanges(limit = 10) {
|
|
96
|
+
if (!fs.existsSync(this.logFilePath))
|
|
97
|
+
return [];
|
|
98
|
+
const content = fs.readFileSync(this.logFilePath, 'utf8');
|
|
99
|
+
const entries = content.split('## ').slice(1);
|
|
100
|
+
return entries.slice(-limit).map(entry => this.parseEntry(entry));
|
|
101
|
+
}
|
|
102
|
+
parseEntry(entry) {
|
|
103
|
+
// 简化解析
|
|
104
|
+
const lines = entry.split('\n');
|
|
105
|
+
const change = {
|
|
106
|
+
timestamp: '',
|
|
107
|
+
modifier: '',
|
|
108
|
+
file: '',
|
|
109
|
+
changes: [],
|
|
110
|
+
reason: '',
|
|
111
|
+
verified: false,
|
|
112
|
+
gatewayRestarted: false,
|
|
113
|
+
};
|
|
114
|
+
for (const line of lines) {
|
|
115
|
+
if (line.startsWith('- **修改者**: ')) {
|
|
116
|
+
change.modifier = line.replace('- **修改者**: ', '');
|
|
117
|
+
}
|
|
118
|
+
else if (line.startsWith('- **文件**: ')) {
|
|
119
|
+
change.file = line.replace('- **文件**: ', '');
|
|
120
|
+
}
|
|
121
|
+
else if (line.startsWith('- **原因**: ')) {
|
|
122
|
+
change.reason = line.replace('- **原因**: ', '');
|
|
123
|
+
}
|
|
124
|
+
else if (line.includes('✅ 通过')) {
|
|
125
|
+
change.verified = true;
|
|
126
|
+
}
|
|
127
|
+
else if (line.includes('✅ 是')) {
|
|
128
|
+
change.gatewayRestarted = true;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
return change;
|
|
132
|
+
}
|
|
133
|
+
exportToJson(outputPath) {
|
|
134
|
+
const changes = this.getRecentChanges(100);
|
|
135
|
+
fs.writeFileSync(outputPath, JSON.stringify(changes, null, 2));
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
exports.ConfigLogger = ConfigLogger;
|
|
139
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config-logger.js","sourceRoot":"","sources":["../../../src/core/loggers/config-logger.ts"],"names":[],"mappings":"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","sourcesContent":["// src/loggers/config-logger.ts - 配置变更日志\n\nimport * as fs from 'fs';\nimport * as path from 'path';\n\nexport interface ConfigChange {\n  timestamp: string;\n  modifier: string;         // 修改者（如 \"蓝山\"）\n  file: string;             // 修改的文件\n  changes: Array<{\n    field: string;\n    oldValue: any;\n    newValue: any;\n  }>;\n  reason: string;           // 变更原因\n  verified: boolean;        // 是否验证通过\n  gatewayRestarted: boolean; // 是否重启 Gateway\n  sessionId?: string;       // 会话 ID\n}\n\nexport class ConfigLogger {\n  private logFilePath: string;\n\n  constructor(private workspacePath: string) {\n    const logsDir = path.join(workspacePath, 'logs');\n    if (!fs.existsSync(logsDir)) {\n      fs.mkdirSync(logsDir, { recursive: true });\n    }\n    this.logFilePath = path.join(logsDir, 'config-changes.md');\n    \n    // 初始化日志文件（如果不存在）\n    if (!fs.existsSync(this.logFilePath)) {\n      fs.writeFileSync(this.logFilePath, '# 配置变更日志\\n\\n');\n    }\n  }\n\n  log(change: Omit<ConfigChange, 'timestamp'>): void {\n    const fullChange: ConfigChange = {\n      ...change,\n      timestamp: new Date().toISOString(),\n    };\n\n    const entry = this.formatEntry(fullChange);\n    fs.appendFileSync(this.logFilePath, entry);\n  }\n\n  private formatEntry(change: ConfigChange): string {\n    const date = new Date(change.timestamp);\n    const dateStr = date.toISOString().replace('T', ' ').substring(0, 19);\n\n    let entry = `## ${dateStr}\\n`;\n    entry += `- **修改者**: ${change.modifier}\\n`;\n    entry += `- **文件**: ${change.file}\\n`;\n    entry += `- **原因**: ${change.reason}\\n`;\n    entry += `- **验证**: ${change.verified ? '✅ 通过' : '❌ 失败'}\\n`;\n    entry += `- **Gateway 重启**: ${change.gatewayRestarted ? '✅ 是' : '❌ 否'}\\n`;\n    if (change.sessionId) {\n      entry += `- **会话 ID**: ${change.sessionId}\\n`;\n    }\n    entry += `\\n### 变更详情\\n`;\n    \n    if (change.changes.length === 0) {\n      entry += '_无具体变更记录_\\n';\n    } else {\n      change.changes.forEach((c, i) => {\n        entry += `${i + 1}. \\`${c.field}\\`\\n`;\n        entry += `   - 旧值：\\`${this.truncateValue(c.oldValue)}\\`\\n`;\n        entry += `   - 新值：\\`${this.truncateValue(c.newValue)}\\`\\n`;\n      });\n    }\n\n    entry += `\\n---\\n\\n`;\n    return entry;\n  }\n\n  private truncateValue(value: any, maxLength: number = 100): string {\n    const str = JSON.stringify(value);\n    if (str.length <= maxLength) return str;\n    return str.substring(0, maxLength) + '...';\n  }\n\n  getRecentChanges(limit: number = 10): ConfigChange[] {\n    if (!fs.existsSync(this.logFilePath)) return [];\n    \n    const content = fs.readFileSync(this.logFilePath, 'utf8');\n    const entries = content.split('## ').slice(1);\n    return entries.slice(-limit).map(entry => this.parseEntry(entry));\n  }\n\n  private parseEntry(entry: string): ConfigChange {\n    // 简化解析\n    const lines = entry.split('\\n');\n    const change: ConfigChange = {\n      timestamp: '',\n      modifier: '',\n      file: '',\n      changes: [],\n      reason: '',\n      verified: false,\n      gatewayRestarted: false,\n    };\n\n    for (const line of lines) {\n      if (line.startsWith('- **修改者**: ')) {\n        change.modifier = line.replace('- **修改者**: ', '');\n      } else if (line.startsWith('- **文件**: ')) {\n        change.file = line.replace('- **文件**: ', '');\n      } else if (line.startsWith('- **原因**: ')) {\n        change.reason = line.replace('- **原因**: ', '');\n      } else if (line.includes('✅ 通过')) {\n        change.verified = true;\n      } else if (line.includes('✅ 是')) {\n        change.gatewayRestarted = true;\n      }\n    }\n\n    return change;\n  }\n\n  exportToJson(outputPath: string): void {\n    const changes = this.getRecentChanges(100);\n    fs.writeFileSync(outputPath, JSON.stringify(changes, null, 2));\n  }\n}\n"]}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.OperationLogger = exports.ConfigLogger = void 0;
|
|
4
|
+
// src/loggers/index.ts
|
|
5
|
+
var config_logger_1 = require("./config-logger");
|
|
6
|
+
Object.defineProperty(exports, "ConfigLogger", { enumerable: true, get: function () { return config_logger_1.ConfigLogger; } });
|
|
7
|
+
var operation_logger_1 = require("./operation-logger");
|
|
8
|
+
Object.defineProperty(exports, "OperationLogger", { enumerable: true, get: function () { return operation_logger_1.OperationLogger; } });
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29yZS9sb2dnZXJzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHVCQUF1QjtBQUN2QixpREFBK0M7QUFBdEMsNkdBQUEsWUFBWSxPQUFBO0FBRXJCLHVEQUFxRDtBQUE1QyxtSEFBQSxlQUFlLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBzcmMvbG9nZ2Vycy9pbmRleC50c1xuZXhwb3J0IHsgQ29uZmlnTG9nZ2VyIH0gZnJvbSAnLi9jb25maWctbG9nZ2VyJztcbmV4cG9ydCB0eXBlIHsgQ29uZmlnQ2hhbmdlIH0gZnJvbSAnLi9jb25maWctbG9nZ2VyJztcbmV4cG9ydCB7IE9wZXJhdGlvbkxvZ2dlciB9IGZyb20gJy4vb3BlcmF0aW9uLWxvZ2dlcic7XG5leHBvcnQgdHlwZSB7IE9wZXJhdGlvbkVudHJ5IH0gZnJvbSAnLi9vcGVyYXRpb24tbG9nZ2VyJztcbiJdfQ==
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
export interface OperationEntry {
|
|
2
|
+
timestamp: string;
|
|
3
|
+
sessionId: string;
|
|
4
|
+
modifier: string;
|
|
5
|
+
type: 'config_change' | 'file_edit' | 'file_delete' | 'system_command' | 'gateway_restart' | 'other';
|
|
6
|
+
description: string;
|
|
7
|
+
file?: string;
|
|
8
|
+
command?: string;
|
|
9
|
+
result: 'success' | 'error' | 'pending';
|
|
10
|
+
error?: string;
|
|
11
|
+
}
|
|
12
|
+
export declare class OperationLogger {
|
|
13
|
+
private workspacePath;
|
|
14
|
+
private logFilePath;
|
|
15
|
+
constructor(workspacePath: string);
|
|
16
|
+
log(entry: Omit<OperationEntry, 'timestamp'>): void;
|
|
17
|
+
private formatEntry;
|
|
18
|
+
private getTypeIcon;
|
|
19
|
+
getRecentOperations(limit?: number): OperationEntry[];
|
|
20
|
+
private parseEntry;
|
|
21
|
+
getOperationsBySession(sessionId: string, limit?: number): OperationEntry[];
|
|
22
|
+
getOperationsByType(type: OperationEntry['type'], limit?: number): OperationEntry[];
|
|
23
|
+
}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// src/loggers/operation-logger.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.OperationLogger = void 0;
|
|
38
|
+
const fs = __importStar(require("fs"));
|
|
39
|
+
const path = __importStar(require("path"));
|
|
40
|
+
class OperationLogger {
|
|
41
|
+
workspacePath;
|
|
42
|
+
logFilePath;
|
|
43
|
+
constructor(workspacePath) {
|
|
44
|
+
this.workspacePath = workspacePath;
|
|
45
|
+
const logsDir = path.join(workspacePath, 'logs');
|
|
46
|
+
if (!fs.existsSync(logsDir)) {
|
|
47
|
+
fs.mkdirSync(logsDir, { recursive: true });
|
|
48
|
+
}
|
|
49
|
+
this.logFilePath = path.join(logsDir, 'operations.md');
|
|
50
|
+
// 初始化日志文件(如果不存在)
|
|
51
|
+
if (!fs.existsSync(this.logFilePath)) {
|
|
52
|
+
fs.writeFileSync(this.logFilePath, '# 操作日志\n\n');
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
log(entry) {
|
|
56
|
+
const fullEntry = {
|
|
57
|
+
...entry,
|
|
58
|
+
timestamp: new Date().toISOString(),
|
|
59
|
+
};
|
|
60
|
+
const logEntry = this.formatEntry(fullEntry);
|
|
61
|
+
fs.appendFileSync(this.logFilePath, logEntry);
|
|
62
|
+
}
|
|
63
|
+
formatEntry(entry) {
|
|
64
|
+
const date = new Date(entry.timestamp);
|
|
65
|
+
const dateStr = date.toISOString().replace('T', ' ').substring(0, 19);
|
|
66
|
+
const icon = entry.result === 'success' ? '✅' : entry.result === 'error' ? '❌' : '⏳';
|
|
67
|
+
const typeIcon = this.getTypeIcon(entry.type);
|
|
68
|
+
let logEntry = `### ${dateStr} ${icon}\n\n`;
|
|
69
|
+
logEntry += `- **类型**: ${typeIcon} ${entry.type}\n`;
|
|
70
|
+
logEntry += `- **会话**: ${entry.sessionId}\n`;
|
|
71
|
+
logEntry += `- **操作者**: ${entry.modifier}\n`;
|
|
72
|
+
logEntry += `- **描述**: ${entry.description}\n`;
|
|
73
|
+
if (entry.file) {
|
|
74
|
+
logEntry += `- **文件**: \`${entry.file}\`\n`;
|
|
75
|
+
}
|
|
76
|
+
if (entry.command) {
|
|
77
|
+
logEntry += `- **命令**: \`${entry.command}\`\n`;
|
|
78
|
+
}
|
|
79
|
+
if (entry.error) {
|
|
80
|
+
logEntry += `- **错误**: ${entry.error}\n`;
|
|
81
|
+
}
|
|
82
|
+
logEntry += `\n`;
|
|
83
|
+
return logEntry;
|
|
84
|
+
}
|
|
85
|
+
getTypeIcon(type) {
|
|
86
|
+
const icons = {
|
|
87
|
+
config_change: '⚙️',
|
|
88
|
+
file_edit: '📝',
|
|
89
|
+
file_delete: '🗑️',
|
|
90
|
+
system_command: '💻',
|
|
91
|
+
gateway_restart: '🔄',
|
|
92
|
+
other: '📌',
|
|
93
|
+
};
|
|
94
|
+
return icons[type] || '📌';
|
|
95
|
+
}
|
|
96
|
+
getRecentOperations(limit = 20) {
|
|
97
|
+
if (!fs.existsSync(this.logFilePath))
|
|
98
|
+
return [];
|
|
99
|
+
const content = fs.readFileSync(this.logFilePath, 'utf8');
|
|
100
|
+
// 简化解析
|
|
101
|
+
const entries = content.split('### ').slice(1);
|
|
102
|
+
return entries.slice(-limit).map(entry => this.parseEntry(entry));
|
|
103
|
+
}
|
|
104
|
+
parseEntry(entry) {
|
|
105
|
+
// 简化解析
|
|
106
|
+
return {
|
|
107
|
+
timestamp: '',
|
|
108
|
+
sessionId: '',
|
|
109
|
+
modifier: '',
|
|
110
|
+
type: 'other',
|
|
111
|
+
description: '',
|
|
112
|
+
result: 'pending',
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
getOperationsBySession(sessionId, limit = 50) {
|
|
116
|
+
const all = this.getRecentOperations(limit);
|
|
117
|
+
return all.filter(op => op.sessionId === sessionId);
|
|
118
|
+
}
|
|
119
|
+
getOperationsByType(type, limit = 20) {
|
|
120
|
+
const all = this.getRecentOperations(limit);
|
|
121
|
+
return all.filter(op => op.type === type);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
exports.OperationLogger = OperationLogger;
|
|
125
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"operation-logger.js","sourceRoot":"","sources":["../../../src/core/loggers/operation-logger.ts"],"names":[],"mappings":"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","sourcesContent":["// src/loggers/operation-logger.ts - 操作日志\n\nimport * as fs from 'fs';\nimport * as path from 'path';\n\nexport interface OperationEntry {\n  timestamp: string;\n  sessionId: string;\n  modifier: string;\n  type: 'config_change' | 'file_edit' | 'file_delete' | 'system_command' | 'gateway_restart' | 'other';\n  description: string;\n  file?: string;\n  command?: string;\n  result: 'success' | 'error' | 'pending';\n  error?: string;\n}\n\nexport class OperationLogger {\n  private logFilePath: string;\n\n  constructor(private workspacePath: string) {\n    const logsDir = path.join(workspacePath, 'logs');\n    if (!fs.existsSync(logsDir)) {\n      fs.mkdirSync(logsDir, { recursive: true });\n    }\n    this.logFilePath = path.join(logsDir, 'operations.md');\n    \n    // 初始化日志文件（如果不存在）\n    if (!fs.existsSync(this.logFilePath)) {\n      fs.writeFileSync(this.logFilePath, '# 操作日志\\n\\n');\n    }\n  }\n\n  log(entry: Omit<OperationEntry, 'timestamp'>): void {\n    const fullEntry: OperationEntry = {\n      ...entry,\n      timestamp: new Date().toISOString(),\n    };\n\n    const logEntry = this.formatEntry(fullEntry);\n    fs.appendFileSync(this.logFilePath, logEntry);\n  }\n\n  private formatEntry(entry: OperationEntry): string {\n    const date = new Date(entry.timestamp);\n    const dateStr = date.toISOString().replace('T', ' ').substring(0, 19);\n    \n    const icon = entry.result === 'success' ? '✅' : entry.result === 'error' ? '❌' : '⏳';\n    const typeIcon = this.getTypeIcon(entry.type);\n\n    let logEntry = `### ${dateStr} ${icon}\\n\\n`;\n    logEntry += `- **类型**: ${typeIcon} ${entry.type}\\n`;\n    logEntry += `- **会话**: ${entry.sessionId}\\n`;\n    logEntry += `- **操作者**: ${entry.modifier}\\n`;\n    logEntry += `- **描述**: ${entry.description}\\n`;\n    \n    if (entry.file) {\n      logEntry += `- **文件**: \\`${entry.file}\\`\\n`;\n    }\n    \n    if (entry.command) {\n      logEntry += `- **命令**: \\`${entry.command}\\`\\n`;\n    }\n\n    if (entry.error) {\n      logEntry += `- **错误**: ${entry.error}\\n`;\n    }\n\n    logEntry += `\\n`;\n    return logEntry;\n  }\n\n  private getTypeIcon(type: OperationEntry['type']): string {\n    const icons: Record<OperationEntry['type'], string> = {\n      config_change: '⚙️',\n      file_edit: '📝',\n      file_delete: '🗑️',\n      system_command: '💻',\n      gateway_restart: '🔄',\n      other: '📌',\n    };\n    return icons[type] || '📌';\n  }\n\n  getRecentOperations(limit: number = 20): OperationEntry[] {\n    if (!fs.existsSync(this.logFilePath)) return [];\n    \n    const content = fs.readFileSync(this.logFilePath, 'utf8');\n    // 简化解析\n    const entries = content.split('### ').slice(1);\n    return entries.slice(-limit).map(entry => this.parseEntry(entry));\n  }\n\n  private parseEntry(entry: string): OperationEntry {\n    // 简化解析\n    return {\n      timestamp: '',\n      sessionId: '',\n      modifier: '',\n      type: 'other',\n      description: '',\n      result: 'pending',\n    };\n  }\n\n  getOperationsBySession(sessionId: string, limit: number = 50): OperationEntry[] {\n    const all = this.getRecentOperations(limit);\n    return all.filter(op => op.sessionId === sessionId);\n  }\n\n  getOperationsByType(type: OperationEntry['type'], limit: number = 20): OperationEntry[] {\n    const all = this.getRecentOperations(limit);\n    return all.filter(op => op.type === type);\n  }\n}\n"]}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
export interface ContextInjection {
|
|
2
|
+
section: string;
|
|
3
|
+
content: string;
|
|
4
|
+
priority?: 'high' | 'medium' | 'low';
|
|
5
|
+
}
|
|
6
|
+
export declare class ContextInjector {
|
|
7
|
+
private workspacePath;
|
|
8
|
+
constructor(workspacePath: string);
|
|
9
|
+
/**
|
|
10
|
+
* 获取当前工作环境的上下文信息
|
|
11
|
+
*/
|
|
12
|
+
getEnvironmentContext(): ContextInjection[];
|
|
13
|
+
/**
|
|
14
|
+
* 获取目录结构(只显示 Markdown 和配置文件)
|
|
15
|
+
*/
|
|
16
|
+
private getDirectoryStructure;
|
|
17
|
+
/**
|
|
18
|
+
* 获取可用的全局命令
|
|
19
|
+
*/
|
|
20
|
+
private getAvailableCommands;
|
|
21
|
+
/**
|
|
22
|
+
* 格式化为可注入的上下文字符串
|
|
23
|
+
*/
|
|
24
|
+
formatForInjection(contexts?: ContextInjection[]): string;
|
|
25
|
+
}
|
|
@@ -0,0 +1,174 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// src/middleware/context-injector.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.ContextInjector = void 0;
|
|
38
|
+
const fs = __importStar(require("fs"));
|
|
39
|
+
const path = __importStar(require("path"));
|
|
40
|
+
const child_process_1 = require("child_process");
|
|
41
|
+
class ContextInjector {
|
|
42
|
+
workspacePath;
|
|
43
|
+
constructor(workspacePath) {
|
|
44
|
+
this.workspacePath = workspacePath;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* 获取当前工作环境的上下文信息
|
|
48
|
+
*/
|
|
49
|
+
getEnvironmentContext() {
|
|
50
|
+
const injections = [];
|
|
51
|
+
// 工作目录
|
|
52
|
+
injections.push({
|
|
53
|
+
section: '当前工作目录',
|
|
54
|
+
content: this.workspacePath,
|
|
55
|
+
priority: 'high',
|
|
56
|
+
});
|
|
57
|
+
// 目录结构
|
|
58
|
+
try {
|
|
59
|
+
const dirStructure = this.getDirectoryStructure();
|
|
60
|
+
injections.push({
|
|
61
|
+
section: '项目结构',
|
|
62
|
+
content: dirStructure,
|
|
63
|
+
priority: 'medium',
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
catch (error) {
|
|
67
|
+
// 忽略错误
|
|
68
|
+
}
|
|
69
|
+
// Node.js 版本
|
|
70
|
+
try {
|
|
71
|
+
const nodeVersion = (0, child_process_1.execSync)('node --version').toString().trim();
|
|
72
|
+
injections.push({
|
|
73
|
+
section: 'Node.js 环境',
|
|
74
|
+
content: `Node.js: ${nodeVersion}`,
|
|
75
|
+
priority: 'medium',
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
catch (error) {
|
|
79
|
+
// 忽略错误
|
|
80
|
+
}
|
|
81
|
+
// 可用的全局命令
|
|
82
|
+
try {
|
|
83
|
+
const commands = this.getAvailableCommands();
|
|
84
|
+
injections.push({
|
|
85
|
+
section: '可用命令',
|
|
86
|
+
content: commands,
|
|
87
|
+
priority: 'low',
|
|
88
|
+
});
|
|
89
|
+
}
|
|
90
|
+
catch (error) {
|
|
91
|
+
// 忽略错误
|
|
92
|
+
}
|
|
93
|
+
return injections;
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* 获取目录结构(只显示 Markdown 和配置文件)
|
|
97
|
+
*/
|
|
98
|
+
getDirectoryStructure() {
|
|
99
|
+
const maxDepth = 3;
|
|
100
|
+
const result = [];
|
|
101
|
+
const scanDir = (dir, depth, prefix = '') => {
|
|
102
|
+
if (depth > maxDepth)
|
|
103
|
+
return;
|
|
104
|
+
let items;
|
|
105
|
+
try {
|
|
106
|
+
items = fs.readdirSync(dir);
|
|
107
|
+
}
|
|
108
|
+
catch (error) {
|
|
109
|
+
return;
|
|
110
|
+
}
|
|
111
|
+
// 过滤:只显示 .md, .json, .js, .ts 文件,跳过 node_modules, .git, dist
|
|
112
|
+
const filtered = items.filter(item => {
|
|
113
|
+
if (['node_modules', '.git', 'dist', '.DS_Store'].includes(item))
|
|
114
|
+
return false;
|
|
115
|
+
const ext = path.extname(item);
|
|
116
|
+
const isRelevant = ['.md', '.json', '.js', '.ts', '.sh'].includes(ext);
|
|
117
|
+
const isDir = fs.statSync(path.join(dir, item)).isDirectory();
|
|
118
|
+
return isRelevant || isDir;
|
|
119
|
+
});
|
|
120
|
+
for (const item of filtered) {
|
|
121
|
+
const fullPath = path.join(dir, item);
|
|
122
|
+
const stat = fs.statSync(fullPath);
|
|
123
|
+
if (stat.isDirectory()) {
|
|
124
|
+
result.push(`${prefix}${item}/`);
|
|
125
|
+
scanDir(fullPath, depth + 1, prefix + ' ');
|
|
126
|
+
}
|
|
127
|
+
else {
|
|
128
|
+
result.push(`${prefix}${item}`);
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
};
|
|
132
|
+
scanDir(this.workspacePath, 0);
|
|
133
|
+
return result.join('\n');
|
|
134
|
+
}
|
|
135
|
+
/**
|
|
136
|
+
* 获取可用的全局命令
|
|
137
|
+
*/
|
|
138
|
+
getAvailableCommands() {
|
|
139
|
+
const commands = [
|
|
140
|
+
'node', 'npm', 'npx',
|
|
141
|
+
'git',
|
|
142
|
+
'curl', 'wget',
|
|
143
|
+
'jq',
|
|
144
|
+
'openclaw',
|
|
145
|
+
];
|
|
146
|
+
const available = [];
|
|
147
|
+
for (const cmd of commands) {
|
|
148
|
+
try {
|
|
149
|
+
const version = (0, child_process_1.execSync)(`${cmd} --version`, { stdio: ['pipe', 'pipe', 'ignore'] })
|
|
150
|
+
.toString()
|
|
151
|
+
.trim()
|
|
152
|
+
.split('\n')[0];
|
|
153
|
+
available.push(`${cmd}: ${version}`);
|
|
154
|
+
}
|
|
155
|
+
catch (error) {
|
|
156
|
+
// 命令不可用
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
return available.join('\n') || '无可用命令';
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* 格式化为可注入的上下文字符串
|
|
163
|
+
*/
|
|
164
|
+
formatForInjection(contexts) {
|
|
165
|
+
const ctx = contexts || this.getEnvironmentContext();
|
|
166
|
+
let output = '## 当前环境上下文\n\n';
|
|
167
|
+
ctx.forEach(c => {
|
|
168
|
+
output += `### ${c.section}\n\`\`\`\n${c.content}\n\`\`\`\n\n`;
|
|
169
|
+
});
|
|
170
|
+
return output;
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
exports.ContextInjector = ContextInjector;
|
|
174
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-injector.js","sourceRoot":"","sources":["../../../src/core/middleware/context-injector.ts"],"names":[],"mappings":"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","sourcesContent":["// src/middleware/context-injector.ts - 环境上下文注入器\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { execSync } from 'child_process';\n\nexport interface ContextInjection {\n  section: string;\n  content: string;\n  priority?: 'high' | 'medium' | 'low';\n}\n\nexport class ContextInjector {\n  constructor(private workspacePath: string) {}\n\n  /**\n   * 获取当前工作环境的上下文信息\n   */\n  getEnvironmentContext(): ContextInjection[] {\n    const injections: ContextInjection[] = [];\n\n    // 工作目录\n    injections.push({\n      section: '当前工作目录',\n      content: this.workspacePath,\n      priority: 'high',\n    });\n\n    // 目录结构\n    try {\n      const dirStructure = this.getDirectoryStructure();\n      injections.push({\n        section: '项目结构',\n        content: dirStructure,\n        priority: 'medium',\n      });\n    } catch (error) {\n      // 忽略错误\n    }\n\n    // Node.js 版本\n    try {\n      const nodeVersion = execSync('node --version').toString().trim();\n      injections.push({\n        section: 'Node.js 环境',\n        content: `Node.js: ${nodeVersion}`,\n        priority: 'medium',\n      });\n    } catch (error) {\n      // 忽略错误\n    }\n\n    // 可用的全局命令\n    try {\n      const commands = this.getAvailableCommands();\n      injections.push({\n        section: '可用命令',\n        content: commands,\n        priority: 'low',\n      });\n    } catch (error) {\n      // 忽略错误\n    }\n\n    return injections;\n  }\n\n  /**\n   * 获取目录结构（只显示 Markdown 和配置文件）\n   */\n  private getDirectoryStructure(): string {\n    const maxDepth = 3;\n    const result: string[] = [];\n\n    const scanDir = (dir: string, depth: number, prefix: string = ''): void => {\n      if (depth > maxDepth) return;\n\n      let items: string[];\n      try {\n        items = fs.readdirSync(dir);\n      } catch (error) {\n        return;\n      }\n\n      // 过滤：只显示 .md, .json, .js, .ts 文件，跳过 node_modules, .git, dist\n      const filtered = items.filter(item => {\n        if (['node_modules', '.git', 'dist', '.DS_Store'].includes(item)) return false;\n        const ext = path.extname(item);\n        const isRelevant = ['.md', '.json', '.js', '.ts', '.sh'].includes(ext);\n        const isDir = fs.statSync(path.join(dir, item)).isDirectory();\n        return isRelevant || isDir;\n      });\n\n      for (const item of filtered) {\n        const fullPath = path.join(dir, item);\n        const stat = fs.statSync(fullPath);\n        \n        if (stat.isDirectory()) {\n          result.push(`${prefix}${item}/`);\n          scanDir(fullPath, depth + 1, prefix + '  ');\n        } else {\n          result.push(`${prefix}${item}`);\n        }\n      }\n    };\n\n    scanDir(this.workspacePath, 0);\n    return result.join('\\n');\n  }\n\n  /**\n   * 获取可用的全局命令\n   */\n  private getAvailableCommands(): string {\n    const commands = [\n      'node', 'npm', 'npx',\n      'git',\n      'curl', 'wget',\n      'jq',\n      'openclaw',\n    ];\n\n    const available: string[] = [];\n    for (const cmd of commands) {\n      try {\n        const version = execSync(`${cmd} --version`, { stdio: ['pipe', 'pipe', 'ignore'] })\n          .toString()\n          .trim()\n          .split('\\n')[0];\n        available.push(`${cmd}: ${version}`);\n      } catch (error) {\n        // 命令不可用\n      }\n    }\n\n    return available.join('\\n') || '无可用命令';\n  }\n\n  /**\n   * 格式化为可注入的上下文字符串\n   */\n  formatForInjection(contexts?: ContextInjection[]): string {\n    const ctx = contexts || this.getEnvironmentContext();\n    \n    let output = '## 当前环境上下文\\n\\n';\n    ctx.forEach(c => {\n      output += `### ${c.section}\\n\\`\\`\\`\\n${c.content}\\n\\`\\`\\`\\n\\n`;\n    });\n    \n    return output;\n  }\n}\n"]}
|