@hongmaple0820/scale-engine 0.7.0 → 0.7.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.
- package/dist/adapters/ClaudeCodeAdapter.js +4 -6
- package/dist/adapters/ClaudeCodeAdapter.js.map +1 -1
- package/dist/adapters/CodexAdapter.js +3 -5
- package/dist/adapters/CodexAdapter.js.map +1 -1
- package/dist/adapters/CursorAdapter.js +3 -5
- package/dist/adapters/CursorAdapter.js.map +1 -1
- package/dist/adapters/GeminiAdapter.js +3 -5
- package/dist/adapters/GeminiAdapter.js.map +1 -1
- package/dist/adapters/HermesAdapter.js +3 -5
- package/dist/adapters/HermesAdapter.js.map +1 -1
- package/dist/adapters/OpenClawAdapter.js +3 -5
- package/dist/adapters/OpenClawAdapter.js.map +1 -1
- package/dist/adapters/OpenCodeAdapter.js +3 -5
- package/dist/adapters/OpenCodeAdapter.js.map +1 -1
- package/dist/adapters/QCoderAdapter.js +3 -5
- package/dist/adapters/QCoderAdapter.js.map +1 -1
- package/dist/adapters/TraeAdapter.js +3 -5
- package/dist/adapters/TraeAdapter.js.map +1 -1
- package/dist/adapters/VSCAdapter.js +3 -5
- package/dist/adapters/VSCAdapter.js.map +1 -1
- package/dist/adapters/WorkBuddyAdapter.js +3 -5
- package/dist/adapters/WorkBuddyAdapter.js.map +1 -1
- package/dist/api/cli.js +58 -2
- package/dist/api/cli.js.map +1 -1
- package/dist/api/doctor.js +2 -0
- package/dist/api/doctor.js.map +1 -1
- package/dist/api/mcp.js +5 -0
- package/dist/api/mcp.js.map +1 -1
- package/dist/artifact/fsm.js +5 -3
- package/dist/artifact/fsm.js.map +1 -1
- package/dist/artifact/sqliteStore.js +14 -1
- package/dist/artifact/sqliteStore.js.map +1 -1
- package/dist/artifact/store.js +4 -2
- package/dist/artifact/store.js.map +1 -1
- package/dist/artifact/types.d.ts +1 -1
- package/dist/artifact/types.js +3 -0
- package/dist/artifact/types.js.map +1 -1
- package/dist/context/ContextBuilder.d.ts +4 -0
- package/dist/context/ContextBuilder.js +47 -10
- package/dist/context/ContextBuilder.js.map +1 -1
- package/dist/core/container.js +2 -4
- package/dist/core/container.js.map +1 -1
- package/dist/core/eventBus.js +6 -5
- package/dist/core/eventBus.js.map +1 -1
- package/dist/dashboard/DashboardServer.d.ts +63 -0
- package/dist/dashboard/DashboardServer.js +167 -0
- package/dist/dashboard/DashboardServer.js.map +1 -0
- package/dist/evolution/AutoDefectCreator.d.ts +34 -0
- package/dist/evolution/AutoDefectCreator.js +115 -0
- package/dist/evolution/AutoDefectCreator.js.map +1 -0
- package/dist/evolution/BehaviorTracker.d.ts +8 -0
- package/dist/evolution/BehaviorTracker.js +21 -3
- package/dist/evolution/BehaviorTracker.js.map +1 -1
- package/dist/evolution/EvolutionEngine.js +14 -3
- package/dist/evolution/EvolutionEngine.js.map +1 -1
- package/dist/evolution/EvolutionEvaluator.d.ts +59 -0
- package/dist/evolution/EvolutionEvaluator.js +115 -0
- package/dist/evolution/EvolutionEvaluator.js.map +1 -0
- package/dist/evolution/LessonValidator.d.ts +36 -0
- package/dist/evolution/LessonValidator.js +136 -0
- package/dist/evolution/LessonValidator.js.map +1 -0
- package/dist/fsm/FSMAgentBridge.d.ts +59 -0
- package/dist/fsm/FSMAgentBridge.js +195 -0
- package/dist/fsm/FSMAgentBridge.js.map +1 -0
- package/dist/fsm/index.d.ts +2 -0
- package/dist/fsm/index.js +3 -0
- package/dist/fsm/index.js.map +1 -0
- package/dist/guardrails/DetectorEnhanced.d.ts +111 -0
- package/dist/guardrails/DetectorEnhanced.js +200 -0
- package/dist/guardrails/DetectorEnhanced.js.map +1 -0
- package/dist/guardrails/Gateway.js +7 -6
- package/dist/guardrails/Gateway.js.map +1 -1
- package/dist/guardrails/advancedDetectors.js +31 -33
- package/dist/guardrails/advancedDetectors.js.map +1 -1
- package/dist/guardrails/detectors.js +14 -24
- package/dist/guardrails/detectors.js.map +1 -1
- package/dist/hooks/HookDeployer.d.ts +44 -0
- package/dist/hooks/HookDeployer.js +145 -0
- package/dist/hooks/HookDeployer.js.map +1 -0
- package/dist/hooks/HookGeneratorEnhanced.d.ts +67 -0
- package/dist/hooks/HookGeneratorEnhanced.js +238 -0
- package/dist/hooks/HookGeneratorEnhanced.js.map +1 -0
- package/dist/hooks/index.d.ts +4 -0
- package/dist/hooks/index.js +4 -0
- package/dist/hooks/index.js.map +1 -0
- package/dist/index.d.ts +25 -9
- package/dist/index.js +19 -10
- package/dist/index.js.map +1 -1
- package/dist/knowledge/KnowledgeBase.js +3 -2
- package/dist/knowledge/KnowledgeBase.js.map +1 -1
- package/dist/knowledge/SQLiteKnowledgeBase.js +3 -1
- package/dist/knowledge/SQLiteKnowledgeBase.js.map +1 -1
- package/dist/routing/ModelRouter.js +2 -0
- package/dist/routing/ModelRouter.js.map +1 -1
- package/dist/skills/SkillDiscovery.js +1 -0
- package/dist/skills/SkillDiscovery.js.map +1 -1
- package/dist/skills/SkillExecutor.d.ts +28 -0
- package/dist/skills/SkillExecutor.js +84 -0
- package/dist/skills/SkillExecutor.js.map +1 -0
- package/dist/skills/SkillRegistry.d.ts +93 -0
- package/dist/skills/SkillRegistry.js +130 -0
- package/dist/skills/SkillRegistry.js.map +1 -0
- package/dist/skills/TriggerEngine.d.ts +43 -0
- package/dist/skills/TriggerEngine.js +144 -0
- package/dist/skills/TriggerEngine.js.map +1 -0
- package/dist/skills/coreSkills.d.ts +6 -0
- package/dist/skills/coreSkills.js +41 -0
- package/dist/skills/coreSkills.js.map +1 -0
- package/dist/skills/index.d.ts +4 -0
- package/dist/skills/index.js +6 -0
- package/dist/skills/index.js.map +1 -0
- package/dist/tasks/TaskEngine.js +5 -1
- package/dist/tasks/TaskEngine.js.map +1 -1
- package/dist/workflows/GateParser.d.ts +55 -0
- package/dist/workflows/GateParser.js +73 -0
- package/dist/workflows/GateParser.js.map +1 -0
- package/dist/workflows/WorkflowExecutor.d.ts +56 -0
- package/dist/workflows/WorkflowExecutor.js +145 -0
- package/dist/workflows/WorkflowExecutor.js.map +1 -0
- package/dist/workflows/index.d.ts +4 -0
- package/dist/workflows/index.js +5 -0
- package/dist/workflows/index.js.map +1 -0
- package/package.json +5 -2
- package/dist/agents/AgentManager.d.ts +0 -14
- package/dist/agents/AgentManager.js +0 -85
- package/dist/agents/AgentManager.js.map +0 -1
- package/dist/agents/IAgent.d.ts +0 -53
- package/dist/agents/IAgent.js +0 -4
- package/dist/agents/IAgent.js.map +0 -1
- package/dist/agents/definitions/debugger.d.ts +0 -2
- package/dist/agents/definitions/debugger.js +0 -6
- package/dist/agents/definitions/debugger.js.map +0 -1
- package/dist/agents/definitions/doc-writer.d.ts +0 -2
- package/dist/agents/definitions/doc-writer.js +0 -6
- package/dist/agents/definitions/doc-writer.js.map +0 -1
- package/dist/agents/definitions/implementer.d.ts +0 -2
- package/dist/agents/definitions/implementer.js +0 -6
- package/dist/agents/definitions/implementer.js.map +0 -1
- package/dist/agents/definitions/planner.d.ts +0 -2
- package/dist/agents/definitions/planner.js +0 -6
- package/dist/agents/definitions/planner.js.map +0 -1
- package/dist/agents/definitions/researcher.d.ts +0 -2
- package/dist/agents/definitions/researcher.js +0 -6
- package/dist/agents/definitions/researcher.js.map +0 -1
- package/dist/agents/definitions/reviewer.d.ts +0 -2
- package/dist/agents/definitions/reviewer.js +0 -6
- package/dist/agents/definitions/reviewer.js.map +0 -1
- package/dist/agents/definitions/security.d.ts +0 -2
- package/dist/agents/definitions/security.js +0 -6
- package/dist/agents/definitions/security.js.map +0 -1
- package/dist/agents/definitions/tester.d.ts +0 -2
- package/dist/agents/definitions/tester.js +0 -6
- package/dist/agents/definitions/tester.js.map +0 -1
- package/dist/agents/definitions${file}.d.ts +0 -1
- package/dist/agents/definitions${file}.js +0 -14
- package/dist/agents/definitions${file}.js.map +0 -1
- package/dist/agents/index.d.ts +0 -15
- package/dist/agents/index.js +0 -34
- package/dist/agents/index.js.map +0 -1
- package/dist/dashboard/index.d.ts +0 -2
- package/dist/dashboard/index.js +0 -2
- package/dist/dashboard/index.js.map +0 -1
- package/dist/dashboard/server.d.ts +0 -52
- package/dist/dashboard/server.js +0 -83
- package/dist/dashboard/server.js.map +0 -1
- package/dist/evolution/PatternExtractor.d.ts +0 -40
- package/dist/evolution/PatternExtractor.js +0 -83
- package/dist/evolution/PatternExtractor.js.map +0 -1
- package/dist/evolution/SkillCreator.d.ts +0 -40
- package/dist/evolution/SkillCreator.js +0 -118
- package/dist/evolution/SkillCreator.js.map +0 -1
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
// SCALE Engine — FSM Agent Bridge (v0.7.0)
|
|
2
|
+
// 让 Agent 能感知 FSM 状态约束
|
|
3
|
+
export class FSMAgentBridge {
|
|
4
|
+
fsm;
|
|
5
|
+
store;
|
|
6
|
+
constructor(fsm, store) {
|
|
7
|
+
this.fsm = fsm;
|
|
8
|
+
this.store = store;
|
|
9
|
+
}
|
|
10
|
+
async getFSMContext(artifactId) {
|
|
11
|
+
const artifact = await this.store.get(artifactId);
|
|
12
|
+
if (!artifact)
|
|
13
|
+
return null;
|
|
14
|
+
const fsmDef = this.fsm.getDefinition(artifact.type);
|
|
15
|
+
if (!fsmDef)
|
|
16
|
+
return null;
|
|
17
|
+
// 计算允许的迁移
|
|
18
|
+
const availableActions = await this.fsm.availableActions(artifactId);
|
|
19
|
+
// 计算阻塞原因
|
|
20
|
+
const blockingReasons = [];
|
|
21
|
+
// 找出从当前状态出发的所有迁移
|
|
22
|
+
const transitionsFromCurrent = fsmDef.transitions.filter(t => t.from === artifact.status);
|
|
23
|
+
for (const tx of transitionsFromCurrent) {
|
|
24
|
+
const result = await this.fsm.canTransition(artifactId, tx.action);
|
|
25
|
+
if (!result.allowed && result.blockedBy) {
|
|
26
|
+
blockingReasons.push(`${tx.action}: ${result.blockedBy.map(g => g.message).join(', ')}`);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
// 下游影响
|
|
30
|
+
const downstreamImpact = await this.calculateDownstreamImpact(artifact);
|
|
31
|
+
// 父子状态
|
|
32
|
+
const parentStatus = artifact.parents.length > 0
|
|
33
|
+
? (await this.store.get(artifact.parents[0]))?.status
|
|
34
|
+
: undefined;
|
|
35
|
+
const childrenStatuses = (await this.store.findChildren(artifactId))
|
|
36
|
+
.map(c => c.status);
|
|
37
|
+
return {
|
|
38
|
+
artifactId,
|
|
39
|
+
artifactType: artifact.type,
|
|
40
|
+
currentStatus: artifact.status,
|
|
41
|
+
allowedTransitions: availableActions,
|
|
42
|
+
blockingReasons,
|
|
43
|
+
downstreamImpact,
|
|
44
|
+
parentStatus,
|
|
45
|
+
childrenStatuses,
|
|
46
|
+
};
|
|
47
|
+
}
|
|
48
|
+
async checkOperation(artifactId, operation) {
|
|
49
|
+
const artifact = await this.store.get(artifactId);
|
|
50
|
+
if (!artifact)
|
|
51
|
+
return { allowed: false, reasons: ['Artifact not found'] };
|
|
52
|
+
const fsmDef = this.fsm.getDefinition(artifact.type);
|
|
53
|
+
if (!fsmDef)
|
|
54
|
+
return { allowed: false, reasons: ['No FSM definition for this artifact type'] };
|
|
55
|
+
// 检查是否是合法迁移
|
|
56
|
+
const transitionsFromCurrent = fsmDef.transitions.filter(t => t.from === artifact.status);
|
|
57
|
+
const transition = transitionsFromCurrent.find(t => t.action === operation);
|
|
58
|
+
if (!transition) {
|
|
59
|
+
const allowed = transitionsFromCurrent.map(t => t.action).join(', ') || 'none';
|
|
60
|
+
return {
|
|
61
|
+
allowed: false,
|
|
62
|
+
reasons: [`"${operation}" is not a valid transition from ${artifact.status}. Allowed: ${allowed}`]
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
// 检查 Guards
|
|
66
|
+
const result = await this.fsm.canTransition(artifactId, operation);
|
|
67
|
+
if (!result.allowed && result.blockedBy) {
|
|
68
|
+
return { allowed: false, reasons: result.blockedBy.map(g => g.message) };
|
|
69
|
+
}
|
|
70
|
+
return { allowed: true, reasons: [] };
|
|
71
|
+
}
|
|
72
|
+
async getCreationPrerequisites(artifactType) {
|
|
73
|
+
// 基于 FSM 定义判断创建条件
|
|
74
|
+
const fsmDef = this.fsm.getDefinition(artifactType);
|
|
75
|
+
if (!fsmDef)
|
|
76
|
+
return { requiredParentStatus: [], message: 'No FSM definition available' };
|
|
77
|
+
// 从 FSM 定义中提取依赖关系(简化实现)
|
|
78
|
+
const deps = {
|
|
79
|
+
Plan: { parentType: 'Spec', parentStatus: 'FROZEN' },
|
|
80
|
+
Task: { parentType: 'Plan', parentStatus: 'APPROVED' },
|
|
81
|
+
Change: { parentType: 'Task', parentStatus: 'IN_PROGRESS' },
|
|
82
|
+
Evidence: { parentType: 'Task', parentStatus: 'DONE' },
|
|
83
|
+
};
|
|
84
|
+
const dep = deps[artifactType];
|
|
85
|
+
if (dep) {
|
|
86
|
+
return {
|
|
87
|
+
requiredParentStatus: [`${dep.parentType} must be in ${dep.parentStatus} status`],
|
|
88
|
+
message: `Creating ${artifactType} requires parent ${dep.parentType} to be ${dep.parentStatus}`,
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
return { requiredParentStatus: [], message: 'No special prerequisites' };
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Get FSM context for all artifacts related to a session
|
|
95
|
+
* This is the primary method for SessionStart hook to inject context
|
|
96
|
+
*/
|
|
97
|
+
async getSessionContext(sessionId, eventBus) {
|
|
98
|
+
// Find all artifacts related to this session
|
|
99
|
+
const events = await eventBus.query({
|
|
100
|
+
sessionId,
|
|
101
|
+
types: ['artifact.created', 'artifact.transitioned'],
|
|
102
|
+
limit: 50,
|
|
103
|
+
});
|
|
104
|
+
const artifactIds = new Set();
|
|
105
|
+
for (const event of events) {
|
|
106
|
+
if (event.artifactId)
|
|
107
|
+
artifactIds.add(event.artifactId);
|
|
108
|
+
}
|
|
109
|
+
// Get FSM snapshots for each artifact
|
|
110
|
+
const snapshots = await Promise.all(Array.from(artifactIds).map(id => this.getFSMContext(id)));
|
|
111
|
+
const validSnapshots = snapshots.filter(Boolean);
|
|
112
|
+
// Generate recommendations based on FSM state
|
|
113
|
+
const recommendations = this.generateRecommendations(validSnapshots);
|
|
114
|
+
return {
|
|
115
|
+
sessionId,
|
|
116
|
+
artifacts: validSnapshots,
|
|
117
|
+
recalledLessons: [], // Will be filled by context inject command
|
|
118
|
+
recommendations,
|
|
119
|
+
generatedAt: Date.now(),
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
/**
|
|
123
|
+
* Generate actionable recommendations from FSM state
|
|
124
|
+
*/
|
|
125
|
+
generateRecommendations(snapshots) {
|
|
126
|
+
const recs = [];
|
|
127
|
+
for (const s of snapshots) {
|
|
128
|
+
// Spec needs to be frozen before implementation
|
|
129
|
+
if (s.artifactType === 'Spec' && s.currentStatus === 'REVIEWING') {
|
|
130
|
+
if (s.allowedTransitions.includes('freeze')) {
|
|
131
|
+
recs.push(`Spec ${s.artifactId} is ready to freeze — execute 'scale transition ${s.artifactId} freeze'`);
|
|
132
|
+
}
|
|
133
|
+
else {
|
|
134
|
+
recs.push(`Spec ${s.artifactId} blocked from freezing: ${s.blockingReasons.join('; ')}`);
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
// Plan needs approval before implementation
|
|
138
|
+
if (s.artifactType === 'Plan' && s.currentStatus === 'DRAFT') {
|
|
139
|
+
recs.push(`Plan ${s.artifactId} needs review before implementation`);
|
|
140
|
+
}
|
|
141
|
+
// Task ready to start
|
|
142
|
+
if (s.artifactType === 'Task' && s.currentStatus === 'READY') {
|
|
143
|
+
recs.push(`Task ${s.artifactId} is ready to implement`);
|
|
144
|
+
}
|
|
145
|
+
// Warning for blocked artifacts
|
|
146
|
+
if (s.blockingReasons.length > 0 && s.allowedTransitions.length === 0) {
|
|
147
|
+
recs.push(`⚠️ ${s.artifactId} (${s.artifactType}) is blocked in ${s.currentStatus}`);
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
return recs;
|
|
151
|
+
}
|
|
152
|
+
async injectFSMContextToPrompt(prompt, artifactIds) {
|
|
153
|
+
if (artifactIds.length === 0)
|
|
154
|
+
return prompt;
|
|
155
|
+
const snapshots = await Promise.all(artifactIds.map(id => this.getFSMContext(id)));
|
|
156
|
+
const validSnapshots = snapshots.filter(Boolean);
|
|
157
|
+
if (validSnapshots.length === 0)
|
|
158
|
+
return prompt;
|
|
159
|
+
const fsmBlock = this.formatFSMBlock(validSnapshots);
|
|
160
|
+
// 插入到 prompt 的合适位置
|
|
161
|
+
const insertPoints = ['<context_summary>', '## Task', '## Context', '## 上下文'];
|
|
162
|
+
for (const point of insertPoints) {
|
|
163
|
+
if (prompt.includes(point)) {
|
|
164
|
+
return prompt.replace(point, fsmBlock + '\n' + point);
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
// 如果找不到合适位置,添加到末尾
|
|
168
|
+
return prompt + '\n' + fsmBlock;
|
|
169
|
+
}
|
|
170
|
+
async calculateDownstreamImpact(artifact) {
|
|
171
|
+
const children = await this.store.findChildren(artifact.id);
|
|
172
|
+
return children.map(c => `${c.id} (${c.type}:${c.status})`);
|
|
173
|
+
}
|
|
174
|
+
formatFSMBlock(snapshots) {
|
|
175
|
+
const lines = snapshots.map(s => {
|
|
176
|
+
const allowed = s.allowedTransitions.length > 0 ? s.allowedTransitions.join(', ') : 'none';
|
|
177
|
+
const blocked = s.blockingReasons.length > 0 ? s.blockingReasons.slice(0, 3).join('; ') : 'none';
|
|
178
|
+
const impact = s.downstreamImpact.length > 0 ? `${s.downstreamImpact.length} children` : 'none';
|
|
179
|
+
return `### ${s.artifactId} (${s.artifactType})
|
|
180
|
+
- Status: **${s.currentStatus}**
|
|
181
|
+
- Allowed actions: ${allowed}
|
|
182
|
+
- Blocked: ${blocked}
|
|
183
|
+
- Downstream impact: ${impact}
|
|
184
|
+
- Parent: ${s.parentStatus ?? 'none'}
|
|
185
|
+
- Children: ${s.childrenStatuses.length > 0 ? s.childrenStatuses.join(', ') : 'none'}`;
|
|
186
|
+
});
|
|
187
|
+
return `\n## FSM Status Constraints
|
|
188
|
+
|
|
189
|
+
${lines.join('\n\n')}
|
|
190
|
+
|
|
191
|
+
> **Important**: Only execute actions listed in "Allowed actions". Blocked actions require resolving constraints first.
|
|
192
|
+
`;
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
//# sourceMappingURL=FSMAgentBridge.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"FSMAgentBridge.js","sourceRoot":"","sources":["../../src/fsm/FSMAgentBridge.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/fsm/index.ts"],"names":[],"mappings":"AAAA,qCAAqC;AAErC,OAAO,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAA"}
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
import type { IDetector, DetectorContext } from './Gateway.js';
|
|
2
|
+
import type { ToolUseInput, ToolResultInput, DetectorResult } from '../artifact/types.js';
|
|
3
|
+
import type { IEventBus } from '../core/eventBus.js';
|
|
4
|
+
export interface DetectorTriggerRecord {
|
|
5
|
+
detectorName: string;
|
|
6
|
+
sessionId: string;
|
|
7
|
+
tool: string;
|
|
8
|
+
severity: string;
|
|
9
|
+
triggeredAt: number;
|
|
10
|
+
reason: string;
|
|
11
|
+
}
|
|
12
|
+
export interface DetectorStatistics {
|
|
13
|
+
detectorName: string;
|
|
14
|
+
totalTriggers: number;
|
|
15
|
+
bySeverity: Record<string, number>;
|
|
16
|
+
byTool: Record<string, number>;
|
|
17
|
+
recentTriggers: DetectorTriggerRecord[];
|
|
18
|
+
}
|
|
19
|
+
export interface IDetectorStatisticsTracker {
|
|
20
|
+
record(trigger: DetectorTriggerRecord): void;
|
|
21
|
+
getStats(detectorName: string): DetectorStatistics | null;
|
|
22
|
+
getAllStats(): DetectorStatistics[];
|
|
23
|
+
getRecentTriggers(limit?: number): DetectorTriggerRecord[];
|
|
24
|
+
clear(): void;
|
|
25
|
+
}
|
|
26
|
+
export declare class DetectorStatisticsTracker implements IDetectorStatisticsTracker {
|
|
27
|
+
private triggers;
|
|
28
|
+
private maxRecords;
|
|
29
|
+
constructor(maxRecords?: number);
|
|
30
|
+
record(trigger: DetectorTriggerRecord): void;
|
|
31
|
+
getStats(detectorName: string): DetectorStatistics | null;
|
|
32
|
+
getAllStats(): DetectorStatistics[];
|
|
33
|
+
getRecentTriggers(limit?: number): DetectorTriggerRecord[];
|
|
34
|
+
clear(): void;
|
|
35
|
+
private groupBy;
|
|
36
|
+
}
|
|
37
|
+
export interface DetectorConfig {
|
|
38
|
+
enabled: boolean;
|
|
39
|
+
threshold?: number;
|
|
40
|
+
windowMs?: number;
|
|
41
|
+
customPatterns?: Array<{
|
|
42
|
+
pattern: RegExp;
|
|
43
|
+
description: string;
|
|
44
|
+
}>;
|
|
45
|
+
}
|
|
46
|
+
export interface IDetectorRegistry {
|
|
47
|
+
register(detector: IDetector, hook: 'preTool' | 'postTool' | 'beforeStop', config?: DetectorConfig): void;
|
|
48
|
+
unregister(detectorName: string): boolean;
|
|
49
|
+
getDetector(detectorName: string): IDetector | null;
|
|
50
|
+
getConfig(detectorName: string): DetectorConfig | null;
|
|
51
|
+
setConfig(detectorName: string, config: Partial<DetectorConfig>): void;
|
|
52
|
+
listDetectors(): Array<{
|
|
53
|
+
name: string;
|
|
54
|
+
hook: string;
|
|
55
|
+
enabled: boolean;
|
|
56
|
+
}>;
|
|
57
|
+
enable(detectorName: string): void;
|
|
58
|
+
disable(detectorName: string): void;
|
|
59
|
+
}
|
|
60
|
+
export declare class DetectorRegistry implements IDetectorRegistry {
|
|
61
|
+
private detectors;
|
|
62
|
+
register(detector: IDetector, hook: 'preTool' | 'postTool' | 'beforeStop', config?: DetectorConfig): void;
|
|
63
|
+
unregister(detectorName: string): boolean;
|
|
64
|
+
getDetector(detectorName: string): IDetector | null;
|
|
65
|
+
getConfig(detectorName: string): DetectorConfig | null;
|
|
66
|
+
setConfig(detectorName: string, config: Partial<DetectorConfig>): void;
|
|
67
|
+
listDetectors(): Array<{
|
|
68
|
+
name: string;
|
|
69
|
+
hook: string;
|
|
70
|
+
enabled: boolean;
|
|
71
|
+
}>;
|
|
72
|
+
enable(detectorName: string): void;
|
|
73
|
+
disable(detectorName: string): void;
|
|
74
|
+
}
|
|
75
|
+
export declare class AISlopDetector implements IDetector {
|
|
76
|
+
name: string;
|
|
77
|
+
private patterns;
|
|
78
|
+
private threshold;
|
|
79
|
+
private windowMs;
|
|
80
|
+
constructor(opts?: {
|
|
81
|
+
threshold?: number;
|
|
82
|
+
windowMs?: number;
|
|
83
|
+
});
|
|
84
|
+
check(input: ToolUseInput, ctx: DetectorContext): Promise<DetectorResult>;
|
|
85
|
+
}
|
|
86
|
+
export declare class HallucinationDetector implements IDetector {
|
|
87
|
+
name: string;
|
|
88
|
+
private patterns;
|
|
89
|
+
check(input: ToolResultInput, ctx: DetectorContext): Promise<DetectorResult>;
|
|
90
|
+
}
|
|
91
|
+
export declare class DuplicateEditDetector implements IDetector {
|
|
92
|
+
name: string;
|
|
93
|
+
private maxDuplicates;
|
|
94
|
+
constructor(opts?: {
|
|
95
|
+
maxDuplicates?: number;
|
|
96
|
+
});
|
|
97
|
+
check(input: ToolUseInput, ctx: DetectorContext): Promise<DetectorResult>;
|
|
98
|
+
}
|
|
99
|
+
export interface IEnhancedGateway {
|
|
100
|
+
registry: IDetectorRegistry;
|
|
101
|
+
stats: IDetectorStatisticsTracker;
|
|
102
|
+
}
|
|
103
|
+
export declare class EnhancedGatewayContext implements IEnhancedGateway {
|
|
104
|
+
registry: IDetectorRegistry;
|
|
105
|
+
stats: IDetectorStatisticsTracker;
|
|
106
|
+
constructor(eventBus: IEventBus);
|
|
107
|
+
}
|
|
108
|
+
export declare const ALL_ENHANCED_DETECTORS: Array<{
|
|
109
|
+
detector: IDetector;
|
|
110
|
+
hook: 'preTool' | 'postTool' | 'beforeStop';
|
|
111
|
+
}>;
|
|
@@ -0,0 +1,200 @@
|
|
|
1
|
+
// SCALE Engine — Detector Enhanced (v0.7.0)
|
|
2
|
+
// 增强 Detector 管理:统计、配置、AI-Slop检测、Hallucination检测
|
|
3
|
+
import { logger } from '../core/logger.js';
|
|
4
|
+
export class DetectorStatisticsTracker {
|
|
5
|
+
triggers = [];
|
|
6
|
+
maxRecords;
|
|
7
|
+
constructor(maxRecords = 1000) { this.maxRecords = maxRecords; }
|
|
8
|
+
record(trigger) {
|
|
9
|
+
this.triggers.push(trigger);
|
|
10
|
+
if (this.triggers.length > this.maxRecords)
|
|
11
|
+
this.triggers.shift();
|
|
12
|
+
}
|
|
13
|
+
getStats(detectorName) {
|
|
14
|
+
const matches = this.triggers.filter(t => t.detectorName === detectorName);
|
|
15
|
+
if (matches.length === 0)
|
|
16
|
+
return null;
|
|
17
|
+
return {
|
|
18
|
+
detectorName,
|
|
19
|
+
totalTriggers: matches.length,
|
|
20
|
+
bySeverity: this.groupBy(matches, 'severity'),
|
|
21
|
+
byTool: this.groupBy(matches, 'tool'),
|
|
22
|
+
recentTriggers: matches.slice(-10),
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
getAllStats() {
|
|
26
|
+
const names = new Set(this.triggers.map(t => t.detectorName));
|
|
27
|
+
return Array.from(names).map(n => this.getStats(n)).filter(Boolean);
|
|
28
|
+
}
|
|
29
|
+
getRecentTriggers(limit = 50) { return this.triggers.slice(-limit); }
|
|
30
|
+
clear() { this.triggers = []; }
|
|
31
|
+
groupBy(arr, key) {
|
|
32
|
+
const result = {};
|
|
33
|
+
for (const item of arr) {
|
|
34
|
+
const val = String(item[key]);
|
|
35
|
+
result[val] = (result[val] ?? 0) + 1;
|
|
36
|
+
}
|
|
37
|
+
return result;
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
export class DetectorRegistry {
|
|
41
|
+
detectors = new Map();
|
|
42
|
+
register(detector, hook, config) {
|
|
43
|
+
this.detectors.set(detector.name, { detector, hook, config: config ?? { enabled: true } });
|
|
44
|
+
logger.info({ name: detector.name, hook }, 'Detector registered in registry');
|
|
45
|
+
}
|
|
46
|
+
unregister(detectorName) { return this.detectors.delete(detectorName); }
|
|
47
|
+
getDetector(detectorName) { return this.detectors.get(detectorName)?.detector ?? null; }
|
|
48
|
+
getConfig(detectorName) { return this.detectors.get(detectorName)?.config ?? null; }
|
|
49
|
+
setConfig(detectorName, config) {
|
|
50
|
+
const existing = this.detectors.get(detectorName);
|
|
51
|
+
if (existing)
|
|
52
|
+
existing.config = { ...existing.config, ...config };
|
|
53
|
+
}
|
|
54
|
+
listDetectors() {
|
|
55
|
+
return Array.from(this.detectors.entries()).map(([name, { hook, config }]) => ({ name, hook, enabled: config.enabled }));
|
|
56
|
+
}
|
|
57
|
+
enable(detectorName) { this.setConfig(detectorName, { enabled: true }); }
|
|
58
|
+
disable(detectorName) { this.setConfig(detectorName, { enabled: false }); }
|
|
59
|
+
}
|
|
60
|
+
export class AISlopDetector {
|
|
61
|
+
name = 'ai-slop';
|
|
62
|
+
patterns = [
|
|
63
|
+
{ pattern: /gradient.*purple.*blue/i, description: 'purple-blue gradient overuse' },
|
|
64
|
+
{ pattern: /gradient.*linear.*135deg/i, description: 'generic 135deg gradient' },
|
|
65
|
+
{ pattern: /borderRadius:s*(8|12|16)(px|rem)/i, description: 'uniform rounded corners' },
|
|
66
|
+
{ pattern: /rounded-[(8|12|16)px]/i, description: 'uniform rounded corners (Tailwind)' },
|
|
67
|
+
{ pattern: /hero.*section.*center.*gradient/i, description: 'generic hero section' },
|
|
68
|
+
{ pattern: /grid-cols-3.*gap-4/i, description: 'generic 3-column grid' },
|
|
69
|
+
{ pattern: /flex.*justify-between.*items-center/i, description: 'generic flex center layout' },
|
|
70
|
+
{ pattern: /[🚀💡🎯✨🎉📊🏆⚡💡🔧]/u, description: 'emoji overuse' },
|
|
71
|
+
{ pattern: /background.*blur.*opacity.*0.[1-5]/i, description: 'decorative blur overlay' },
|
|
72
|
+
];
|
|
73
|
+
threshold;
|
|
74
|
+
windowMs;
|
|
75
|
+
constructor(opts = {}) {
|
|
76
|
+
this.threshold = opts.threshold ?? 3;
|
|
77
|
+
this.windowMs = opts.windowMs ?? 5 * 60 * 1000;
|
|
78
|
+
}
|
|
79
|
+
async check(input, ctx) {
|
|
80
|
+
if (!['Edit', 'Write', 'MultiEdit'].includes(input.tool))
|
|
81
|
+
return { triggered: false };
|
|
82
|
+
const content = JSON.stringify(input.args);
|
|
83
|
+
const matches = [];
|
|
84
|
+
for (const { pattern, description } of this.patterns) {
|
|
85
|
+
if (pattern.test(content))
|
|
86
|
+
matches.push(description);
|
|
87
|
+
}
|
|
88
|
+
if (matches.length < 2)
|
|
89
|
+
return { triggered: false };
|
|
90
|
+
const key = 'ai-slop:' + input.sessionId;
|
|
91
|
+
const history = ctx.cache.get(key) ?? [];
|
|
92
|
+
const recent = history.filter(t => Date.now() - t < this.windowMs);
|
|
93
|
+
recent.push(Date.now());
|
|
94
|
+
ctx.cache.set(key, recent);
|
|
95
|
+
if (recent.length >= this.threshold) {
|
|
96
|
+
ctx.eventBus.emit('behavior.ai_slop', { sessionId: input.sessionId, patterns: matches, count: recent.length }, { sessionId: input.sessionId });
|
|
97
|
+
return {
|
|
98
|
+
triggered: true,
|
|
99
|
+
severity: 'warn',
|
|
100
|
+
reason: 'Detected AI-Slop patterns: ' + matches.slice(0, 3).join(', ') + '. Make code look human-written.',
|
|
101
|
+
suggestion: 'Avoid: gradient abuse, uniform rounded corners, emoji, template hero, 3-column grid.',
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
return { triggered: false };
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
export class HallucinationDetector {
|
|
108
|
+
name = 'hallucination';
|
|
109
|
+
patterns = [
|
|
110
|
+
{ pattern: /测试.*通过|passed.*test/i, description: 'unverified test pass claim' },
|
|
111
|
+
{ pattern: /已.*验证|verified.*success/i, description: 'unverified verification claim' },
|
|
112
|
+
{ pattern: /构建.*成功|build.*succeeded/i, description: 'unverified build success claim' },
|
|
113
|
+
{ pattern: /lint.*通过|lint.*passed/i, description: 'unverified lint pass claim' },
|
|
114
|
+
];
|
|
115
|
+
async check(input, ctx) {
|
|
116
|
+
const text = input.output ?? '';
|
|
117
|
+
const matches = [];
|
|
118
|
+
for (const { pattern, description } of this.patterns) {
|
|
119
|
+
if (pattern.test(text))
|
|
120
|
+
matches.push(description);
|
|
121
|
+
}
|
|
122
|
+
if (matches.length === 0)
|
|
123
|
+
return { triggered: false };
|
|
124
|
+
const recentCommands = await ctx.eventBus.query({
|
|
125
|
+
sessionId: input.sessionId,
|
|
126
|
+
types: ['tool.completed'],
|
|
127
|
+
filter: (e) => {
|
|
128
|
+
const p = e.payload;
|
|
129
|
+
return p.tool === 'Bash' && /test|lint|build|verify/i.test(p.args?.command ?? '');
|
|
130
|
+
},
|
|
131
|
+
limit: 10,
|
|
132
|
+
});
|
|
133
|
+
const hasSuccessClaim = matches.some(m => m.includes('unverified'));
|
|
134
|
+
if (hasSuccessClaim && recentCommands.length === 0) {
|
|
135
|
+
ctx.eventBus.emit('behavior.hallucination', { sessionId: input.sessionId, patterns: matches, type: 'unverified_claim' }, { sessionId: input.sessionId });
|
|
136
|
+
return {
|
|
137
|
+
triggered: true,
|
|
138
|
+
severity: 'warn',
|
|
139
|
+
reason: 'Detected hallucination: claiming "' + matches[0] + '" without running verification.',
|
|
140
|
+
suggestion: 'Run actual verification commands: bun test, pnpm lint, pnpm build',
|
|
141
|
+
};
|
|
142
|
+
}
|
|
143
|
+
return { triggered: false };
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
export class DuplicateEditDetector {
|
|
147
|
+
name = 'duplicate-edit';
|
|
148
|
+
maxDuplicates;
|
|
149
|
+
constructor(opts = {}) { this.maxDuplicates = opts.maxDuplicates ?? 2; }
|
|
150
|
+
async check(input, ctx) {
|
|
151
|
+
if (input.tool !== 'Edit')
|
|
152
|
+
return { triggered: false };
|
|
153
|
+
const args = input.args;
|
|
154
|
+
if (!args.old_string || !args.file_path)
|
|
155
|
+
return { triggered: false };
|
|
156
|
+
const key = 'duplicate-edit:' + input.sessionId + ':' + args.file_path;
|
|
157
|
+
const edits = ctx.cache.get(key) ?? [];
|
|
158
|
+
const duplicateCount = edits.filter(s => s === args.old_string).length;
|
|
159
|
+
edits.push(args.old_string);
|
|
160
|
+
ctx.cache.set(key, edits);
|
|
161
|
+
if (duplicateCount >= this.maxDuplicates) {
|
|
162
|
+
ctx.eventBus.emit('behavior.duplicate_edit', { sessionId: input.sessionId, file: args.file_path, count: duplicateCount + 1 }, { sessionId: input.sessionId });
|
|
163
|
+
return {
|
|
164
|
+
triggered: true,
|
|
165
|
+
severity: 'warn',
|
|
166
|
+
reason: 'Detected duplicate edit: same content edited ' + (duplicateCount + 1) + ' times in ' + args.file_path,
|
|
167
|
+
suggestion: 'Check if previous edits applied correctly, try different strategy.',
|
|
168
|
+
};
|
|
169
|
+
}
|
|
170
|
+
return { triggered: false };
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
export class EnhancedGatewayContext {
|
|
174
|
+
registry;
|
|
175
|
+
stats;
|
|
176
|
+
constructor(eventBus) {
|
|
177
|
+
this.registry = new DetectorRegistry();
|
|
178
|
+
this.stats = new DetectorStatisticsTracker();
|
|
179
|
+
// Listen to all events and filter behavior events
|
|
180
|
+
eventBus.on('*', (e) => {
|
|
181
|
+
if (!e.type.startsWith('behavior.'))
|
|
182
|
+
return;
|
|
183
|
+
const payload = e.payload;
|
|
184
|
+
this.stats.record({
|
|
185
|
+
detectorName: String(e.type.replace('behavior.', '')),
|
|
186
|
+
sessionId: String(payload.sessionId ?? 'unknown'),
|
|
187
|
+
tool: String(payload.tool ?? 'unknown'),
|
|
188
|
+
severity: 'warn',
|
|
189
|
+
triggeredAt: Date.now(),
|
|
190
|
+
reason: String(payload.reason ?? ''),
|
|
191
|
+
});
|
|
192
|
+
});
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
export const ALL_ENHANCED_DETECTORS = [
|
|
196
|
+
{ detector: new AISlopDetector(), hook: 'preTool' },
|
|
197
|
+
{ detector: new HallucinationDetector(), hook: 'postTool' },
|
|
198
|
+
{ detector: new DuplicateEditDetector(), hook: 'preTool' },
|
|
199
|
+
];
|
|
200
|
+
//# sourceMappingURL=DetectorEnhanced.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"DetectorEnhanced.js","sourceRoot":"","sources":["../../src/guardrails/DetectorEnhanced.ts"],"names":[],"mappings":"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"}
|
|
@@ -3,14 +3,15 @@
|
|
|
3
3
|
// 设计参考:docs/03-CORE-MODULES.md §3.5
|
|
4
4
|
import { logger } from '../core/logger.js';
|
|
5
5
|
export class Gateway {
|
|
6
|
+
eventBus;
|
|
7
|
+
cache = new Map();
|
|
8
|
+
detectors = {
|
|
9
|
+
preTool: [],
|
|
10
|
+
postTool: [],
|
|
11
|
+
beforeStop: [],
|
|
12
|
+
};
|
|
6
13
|
constructor(eventBus) {
|
|
7
14
|
this.eventBus = eventBus;
|
|
8
|
-
this.cache = new Map();
|
|
9
|
-
this.detectors = {
|
|
10
|
-
preTool: [],
|
|
11
|
-
postTool: [],
|
|
12
|
-
beforeStop: [],
|
|
13
|
-
};
|
|
14
15
|
}
|
|
15
16
|
registerDetector(detector, hook) {
|
|
16
17
|
this.detectors[hook].push(detector);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"Gateway.js","sourceRoot":"","sources":["../../src/guardrails/Gateway.ts"],"names":[],"mappings":"AAAA,8CAA8C;AAC9C,+BAA+B;AAC/B,oCAAoC;AAIpC,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAA;AAmB1C,MAAM,OAAO,OAAO;
|
|
1
|
+
{"version":3,"file":"Gateway.js","sourceRoot":"","sources":["../../src/guardrails/Gateway.ts"],"names":[],"mappings":"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"}
|