@hongmaple0820/scale-engine 0.40.2 → 0.44.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.
- package/README.md +30 -2
- package/dist/api/cli.js +43 -2
- package/dist/api/cli.js.map +1 -1
- package/dist/api/quickstart.d.ts +11 -0
- package/dist/api/quickstart.js +98 -1
- package/dist/api/quickstart.js.map +1 -1
- package/dist/artifact/fsmDefinitions.js +15 -2
- package/dist/artifact/fsmDefinitions.js.map +1 -1
- package/dist/artifact/types.d.ts +1 -1
- package/dist/artifact/types.js.map +1 -1
- package/dist/cache/ScanCache.d.ts +41 -0
- package/dist/cache/ScanCache.js +120 -0
- package/dist/cache/ScanCache.js.map +1 -0
- package/dist/capabilities/BrowserQACapability.d.ts +14 -0
- package/dist/capabilities/BrowserQACapability.js +94 -0
- package/dist/capabilities/BrowserQACapability.js.map +1 -1
- package/dist/cli/autofixCommands.d.ts +22 -0
- package/dist/cli/autofixCommands.js +32 -0
- package/dist/cli/autofixCommands.js.map +1 -0
- package/dist/cli/cortexCommands.d.ts +71 -0
- package/dist/cli/cortexCommands.js +335 -0
- package/dist/cli/cortexCommands.js.map +1 -0
- package/dist/cli/costCommands.d.ts +13 -0
- package/dist/cli/costCommands.js +48 -0
- package/dist/cli/costCommands.js.map +1 -0
- package/dist/cli/orchCommands.d.ts +43 -0
- package/dist/cli/orchCommands.js +135 -0
- package/dist/cli/orchCommands.js.map +1 -0
- package/dist/cli/phaseCommands.js +1 -2
- package/dist/cli/phaseCommands.js.map +1 -1
- package/dist/cli/qaCommands.d.ts +22 -0
- package/dist/cli/qaCommands.js +84 -0
- package/dist/cli/qaCommands.js.map +1 -0
- package/dist/cli/quickstartCommands.d.ts +17 -0
- package/dist/cli/quickstartCommands.js +47 -0
- package/dist/cli/quickstartCommands.js.map +1 -0
- package/dist/cli/shieldCommands.d.ts +30 -0
- package/dist/cli/shieldCommands.js +212 -0
- package/dist/cli/shieldCommands.js.map +1 -0
- package/dist/cli/tuiCommands.d.ts +7 -0
- package/dist/cli/tuiCommands.js +33 -0
- package/dist/cli/tuiCommands.js.map +1 -0
- package/dist/codegraph/CodeIntelligence.d.ts +27 -0
- package/dist/codegraph/CodeIntelligence.js +316 -3
- package/dist/codegraph/CodeIntelligence.js.map +1 -1
- package/dist/config/profiles.js +26 -0
- package/dist/config/profiles.js.map +1 -1
- package/dist/cortex/GovernanceMetrics.d.ts +66 -0
- package/dist/cortex/GovernanceMetrics.js +230 -0
- package/dist/cortex/GovernanceMetrics.js.map +1 -0
- package/dist/cortex/InstinctExtractor.d.ts +61 -0
- package/dist/cortex/InstinctExtractor.js +184 -0
- package/dist/cortex/InstinctExtractor.js.map +1 -0
- package/dist/cortex/InstinctStore.d.ts +54 -0
- package/dist/cortex/InstinctStore.js +266 -0
- package/dist/cortex/InstinctStore.js.map +1 -0
- package/dist/cortex/ReflexionEngine.d.ts +34 -0
- package/dist/cortex/ReflexionEngine.js +157 -0
- package/dist/cortex/ReflexionEngine.js.map +1 -0
- package/dist/cortex/SessionInjector.d.ts +44 -0
- package/dist/cortex/SessionInjector.js +127 -0
- package/dist/cortex/SessionInjector.js.map +1 -0
- package/dist/cortex/adapters/ClaudeAdapter.d.ts +17 -0
- package/dist/cortex/adapters/ClaudeAdapter.js +61 -0
- package/dist/cortex/adapters/ClaudeAdapter.js.map +1 -0
- package/dist/cortex/adapters/CodexAdapter.d.ts +10 -0
- package/dist/cortex/adapters/CodexAdapter.js +52 -0
- package/dist/cortex/adapters/CodexAdapter.js.map +1 -0
- package/dist/cortex/adapters/CursorAdapter.d.ts +10 -0
- package/dist/cortex/adapters/CursorAdapter.js +46 -0
- package/dist/cortex/adapters/CursorAdapter.js.map +1 -0
- package/dist/cortex/adapters/GeminiAdapter.d.ts +11 -0
- package/dist/cortex/adapters/GeminiAdapter.js +48 -0
- package/dist/cortex/adapters/GeminiAdapter.js.map +1 -0
- package/dist/dashboard/DashboardServer.d.ts +33 -13
- package/dist/dashboard/DashboardServer.js +314 -182
- package/dist/dashboard/DashboardServer.js.map +1 -1
- package/dist/dashboard/index.d.ts +2 -2
- package/dist/dashboard/index.js +1 -1
- package/dist/dashboard/index.js.map +1 -1
- package/dist/dashboard/server.d.ts +8 -22
- package/dist/dashboard/server.js +2 -83
- package/dist/dashboard/server.js.map +1 -1
- package/dist/eval/BenchmarkPublisher.d.ts +25 -0
- package/dist/eval/BenchmarkPublisher.js +27 -0
- package/dist/eval/BenchmarkPublisher.js.map +1 -0
- package/dist/guardrails/DependencyAuditor.js +10 -1
- package/dist/guardrails/DependencyAuditor.js.map +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/dist/orchestrator/OrchestratorDaemon.d.ts +44 -0
- package/dist/orchestrator/OrchestratorDaemon.js +150 -0
- package/dist/orchestrator/OrchestratorDaemon.js.map +1 -0
- package/dist/orchestrator/PolicyLoader.d.ts +80 -0
- package/dist/orchestrator/PolicyLoader.js +229 -0
- package/dist/orchestrator/PolicyLoader.js.map +1 -0
- package/dist/orchestrator/ReconciliationLoop.d.ts +71 -0
- package/dist/orchestrator/ReconciliationLoop.js +266 -0
- package/dist/orchestrator/ReconciliationLoop.js.map +1 -0
- package/dist/orchestrator/TrackerAdapter.d.ts +60 -0
- package/dist/orchestrator/TrackerAdapter.js +147 -0
- package/dist/orchestrator/TrackerAdapter.js.map +1 -0
- package/dist/orchestrator/WorkspaceManager.d.ts +66 -0
- package/dist/orchestrator/WorkspaceManager.js +257 -0
- package/dist/orchestrator/WorkspaceManager.js.map +1 -0
- package/dist/qa/BrowserDaemon.d.ts +23 -0
- package/dist/qa/BrowserDaemon.js +79 -0
- package/dist/qa/BrowserDaemon.js.map +1 -0
- package/dist/qa/E2ETestOrchestrator.d.ts +14 -0
- package/dist/qa/E2ETestOrchestrator.js +19 -0
- package/dist/qa/E2ETestOrchestrator.js.map +1 -0
- package/dist/review/CrossModelReviewer.d.ts +35 -0
- package/dist/review/CrossModelReviewer.js +75 -0
- package/dist/review/CrossModelReviewer.js.map +1 -0
- package/dist/review/ReviewAggregator.d.ts +13 -0
- package/dist/review/ReviewAggregator.js +28 -0
- package/dist/review/ReviewAggregator.js.map +1 -0
- package/dist/review/reviewCommands.d.ts +15 -0
- package/dist/review/reviewCommands.js +24 -0
- package/dist/review/reviewCommands.js.map +1 -0
- package/dist/routing/LocalModelProvider.d.ts +11 -0
- package/dist/routing/LocalModelProvider.js +21 -0
- package/dist/routing/LocalModelProvider.js.map +1 -0
- package/dist/routing/ModelRouter.d.ts +12 -0
- package/dist/routing/ModelRouter.js +31 -4
- package/dist/routing/ModelRouter.js.map +1 -1
- package/dist/runtime/AiOsRuntime.d.ts +1 -0
- package/dist/runtime/AiOsRuntime.js +15 -0
- package/dist/runtime/AiOsRuntime.js.map +1 -1
- package/dist/runtime/CostAnalyzer.d.ts +53 -0
- package/dist/runtime/CostAnalyzer.js +160 -0
- package/dist/runtime/CostAnalyzer.js.map +1 -0
- package/dist/runtime/CostOptimizer.d.ts +11 -0
- package/dist/runtime/CostOptimizer.js +21 -0
- package/dist/runtime/CostOptimizer.js.map +1 -0
- package/dist/shield/PolicyCompiler.d.ts +70 -0
- package/dist/shield/PolicyCompiler.js +540 -0
- package/dist/shield/PolicyCompiler.js.map +1 -0
- package/dist/shield/ProtectedPaths.d.ts +39 -0
- package/dist/shield/ProtectedPaths.js +179 -0
- package/dist/shield/ProtectedPaths.js.map +1 -0
- package/dist/shield/ShieldProtocol.d.ts +50 -0
- package/dist/shield/ShieldProtocol.js +103 -0
- package/dist/shield/ShieldProtocol.js.map +1 -0
- package/dist/skills/SkillMdStandard.d.ts +33 -0
- package/dist/skills/SkillMdStandard.js +88 -0
- package/dist/skills/SkillMdStandard.js.map +1 -0
- package/dist/skills/SkillRegistry.d.ts +9 -1
- package/dist/skills/SkillRegistry.js +20 -0
- package/dist/skills/SkillRegistry.js.map +1 -1
- package/dist/skills/interop/GStackInterop.d.ts +15 -0
- package/dist/skills/interop/GStackInterop.js +34 -0
- package/dist/skills/interop/GStackInterop.js.map +1 -0
- package/dist/skills/interop/OMCInterop.d.ts +15 -0
- package/dist/skills/interop/OMCInterop.js +34 -0
- package/dist/skills/interop/OMCInterop.js.map +1 -0
- package/dist/topology/DomainMapper.d.ts +23 -0
- package/dist/topology/DomainMapper.js +179 -0
- package/dist/topology/DomainMapper.js.map +1 -0
- package/dist/topology/LayerClassifier.d.ts +8 -0
- package/dist/topology/LayerClassifier.js +109 -0
- package/dist/topology/LayerClassifier.js.map +1 -0
- package/dist/topology/TourGenerator.d.ts +18 -0
- package/dist/topology/TourGenerator.js +120 -0
- package/dist/topology/TourGenerator.js.map +1 -0
- package/dist/topology/index.d.ts +3 -0
- package/dist/topology/index.js +4 -0
- package/dist/topology/index.js.map +1 -0
- package/dist/tui/TuiDashboard.d.ts +3 -0
- package/dist/tui/TuiDashboard.js +120 -0
- package/dist/tui/TuiDashboard.js.map +1 -0
- package/dist/workflow/GateCatalog.d.ts +2 -0
- package/dist/workflow/GateCatalog.js +59 -3
- package/dist/workflow/GateCatalog.js.map +1 -1
- package/dist/workflow/GovernanceTemplatePacks.d.ts +1 -1
- package/dist/workflow/GovernanceTemplatePacks.js +15 -0
- package/dist/workflow/GovernanceTemplatePacks.js.map +1 -1
- package/dist/workflow/TddLoop.d.ts +2 -0
- package/dist/workflow/TddLoop.js +2 -0
- package/dist/workflow/TddLoop.js.map +1 -1
- package/dist/workflow/UpgradeManager.d.ts +10 -1
- package/dist/workflow/UpgradeManager.js +55 -0
- package/dist/workflow/UpgradeManager.js.map +1 -1
- package/dist/workflow/VerificationProfile.d.ts +8 -0
- package/dist/workflow/VerificationProfile.js +61 -0
- package/dist/workflow/VerificationProfile.js.map +1 -1
- package/dist/workflow/VerificationSchema.d.ts +46 -0
- package/dist/workflow/VerificationSchema.js +97 -0
- package/dist/workflow/VerificationSchema.js.map +1 -0
- package/dist/workflow/autofix/AutoFixEngine.d.ts +37 -0
- package/dist/workflow/autofix/AutoFixEngine.js +169 -0
- package/dist/workflow/autofix/AutoFixEngine.js.map +1 -0
- package/dist/workflow/execution/RalphEngine.d.ts +18 -0
- package/dist/workflow/execution/RalphEngine.js +22 -0
- package/dist/workflow/execution/RalphEngine.js.map +1 -1
- package/dist/workflow/gates/EnhancedGates.d.ts +74 -0
- package/dist/workflow/gates/EnhancedGates.js +653 -0
- package/dist/workflow/gates/EnhancedGates.js.map +1 -0
- package/dist/workflow/gates/GateSystem.d.ts +3 -0
- package/dist/workflow/gates/GateSystem.js +94 -1
- package/dist/workflow/gates/GateSystem.js.map +1 -1
- package/dist/workflow/types.d.ts +1 -1
- package/docs/README.md +3 -0
- package/docs/guides/DEVELOPMENT_WORKFLOW.md +28 -9
- package/docs/guides/GETTING_STARTED.md +19 -0
- package/docs/guides/MIGRATION.md +119 -0
- package/docs/workflow/GATES_AND_SCORE.md +34 -1
- package/docs/workflow/README.md +58 -10
- package/package.json +7 -17
- package/docs/ACTIVE_SECURITY_VISUAL_GATES.md +0 -87
- package/docs/AI_ENGINEERING_OS_POSITIONING.md +0 -607
- package/docs/BACKGROUND_HUNTER.md +0 -62
- package/docs/CODE_INTELLIGENCE.md +0 -180
- package/docs/CONTEXT_BUDGET.md +0 -165
- package/docs/DEPENDENCY_AUDIT.md +0 -118
- package/docs/EVOLUTION_SHADOW_MODE.md +0 -63
- package/docs/GITLAB_FLOW.md +0 -125
- package/docs/GOVERNANCE_DASHBOARD.md +0 -92
- package/docs/MEMORY_BRAIN.md +0 -104
- package/docs/MEMORY_FABRIC.md +0 -161
- package/docs/RESOURCE_GOVERNANCE.md +0 -92
- package/docs/RUNTIME_EVIDENCE.md +0 -101
- package/docs/WORKFLOW_EVAL.md +0 -151
- package/image/wechat-public.jpg +0 -0
- package/image/wxPay.jpg +0 -0
- package/image/zfb.jpg +0 -0
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import { existsSync, mkdirSync, writeFileSync, readFileSync, unlinkSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
import { logger } from '../core/logger.js';
|
|
4
|
+
const DAEMON_DIR = '.scale/qa';
|
|
5
|
+
const PID_FILE = join(DAEMON_DIR, 'daemon.pid');
|
|
6
|
+
const SOCK_FILE = join(DAEMON_DIR, 'daemon.sock');
|
|
7
|
+
const IDLE_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes
|
|
8
|
+
export class BrowserDaemon {
|
|
9
|
+
constructor() {
|
|
10
|
+
this.browser = null;
|
|
11
|
+
this.requestsProcessed = 0;
|
|
12
|
+
this.lastActivity = Date.now();
|
|
13
|
+
this.idleTimer = null;
|
|
14
|
+
}
|
|
15
|
+
async start() {
|
|
16
|
+
if (!existsSync(DAEMON_DIR))
|
|
17
|
+
mkdirSync(DAEMON_DIR, { recursive: true });
|
|
18
|
+
if (existsSync(PID_FILE)) {
|
|
19
|
+
const existingPid = parseInt(readFileSync(PID_FILE, 'utf-8'), 10);
|
|
20
|
+
try {
|
|
21
|
+
process.kill(existingPid, 0);
|
|
22
|
+
throw new Error(`Daemon already running (PID ${existingPid})`);
|
|
23
|
+
}
|
|
24
|
+
catch { }
|
|
25
|
+
}
|
|
26
|
+
// Write PID
|
|
27
|
+
writeFileSync(PID_FILE, String(process.pid));
|
|
28
|
+
// Try to get a browser context
|
|
29
|
+
try {
|
|
30
|
+
// @ts-ignore — optional dependency
|
|
31
|
+
const pw = await import('playwright');
|
|
32
|
+
this.browser = await pw.chromium.launch({ headless: true });
|
|
33
|
+
}
|
|
34
|
+
catch {
|
|
35
|
+
logger.warn('Playwright not installed — daemon will use MCP fallback for each request');
|
|
36
|
+
}
|
|
37
|
+
this.resetIdleTimer();
|
|
38
|
+
logger.info({ pid: process.pid }, 'Browser daemon started');
|
|
39
|
+
return { success: true, pid: process.pid };
|
|
40
|
+
}
|
|
41
|
+
async stop() {
|
|
42
|
+
if (this.idleTimer)
|
|
43
|
+
clearTimeout(this.idleTimer);
|
|
44
|
+
if (this.browser) {
|
|
45
|
+
try {
|
|
46
|
+
await this.browser.close();
|
|
47
|
+
}
|
|
48
|
+
catch { }
|
|
49
|
+
}
|
|
50
|
+
if (existsSync(PID_FILE))
|
|
51
|
+
unlinkSync(PID_FILE);
|
|
52
|
+
logger.info('Browser daemon stopped');
|
|
53
|
+
return { success: true };
|
|
54
|
+
}
|
|
55
|
+
status() {
|
|
56
|
+
return {
|
|
57
|
+
running: existsSync(PID_FILE),
|
|
58
|
+
pid: existsSync(PID_FILE) ? parseInt(readFileSync(PID_FILE, 'utf-8'), 10) : undefined,
|
|
59
|
+
startedAt: undefined,
|
|
60
|
+
requestsProcessed: this.requestsProcessed,
|
|
61
|
+
idleSeconds: Math.floor((Date.now() - this.lastActivity) / 1000),
|
|
62
|
+
};
|
|
63
|
+
}
|
|
64
|
+
getBrowser() {
|
|
65
|
+
this.requestsProcessed++;
|
|
66
|
+
this.lastActivity = Date.now();
|
|
67
|
+
this.resetIdleTimer();
|
|
68
|
+
return this.browser;
|
|
69
|
+
}
|
|
70
|
+
resetIdleTimer() {
|
|
71
|
+
if (this.idleTimer)
|
|
72
|
+
clearTimeout(this.idleTimer);
|
|
73
|
+
this.idleTimer = setTimeout(() => {
|
|
74
|
+
logger.info('Browser daemon idle timeout — shutting down');
|
|
75
|
+
void this.stop();
|
|
76
|
+
}, IDLE_TIMEOUT_MS);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
//# sourceMappingURL=BrowserDaemon.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"BrowserDaemon.js","sourceRoot":"","sources":["../../src/qa/BrowserDaemon.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
export interface DomainSkill {
|
|
2
|
+
domain: string;
|
|
3
|
+
selectors: Record<string, string>;
|
|
4
|
+
flows: Array<{
|
|
5
|
+
name: string;
|
|
6
|
+
steps: unknown[];
|
|
7
|
+
}>;
|
|
8
|
+
savedAt: string;
|
|
9
|
+
}
|
|
10
|
+
export declare function saveDomainSkill(domain: string, selectors: Record<string, string>, flows: Array<{
|
|
11
|
+
name: string;
|
|
12
|
+
steps: unknown[];
|
|
13
|
+
}>): void;
|
|
14
|
+
export declare function loadDomainSkill(domain: string): DomainSkill | null;
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { existsSync, mkdirSync, writeFileSync, readFileSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
export function saveDomainSkill(domain, selectors, flows) {
|
|
4
|
+
const dir = join(process.cwd(), '.scale', 'qa', 'site-skills');
|
|
5
|
+
if (!existsSync(dir))
|
|
6
|
+
mkdirSync(dir, { recursive: true });
|
|
7
|
+
const hash = Buffer.from(domain).toString('hex').slice(0, 16);
|
|
8
|
+
const skill = { domain, selectors, flows, savedAt: new Date().toISOString() };
|
|
9
|
+
writeFileSync(join(dir, `${hash}.json`), JSON.stringify(skill, null, 2));
|
|
10
|
+
}
|
|
11
|
+
export function loadDomainSkill(domain) {
|
|
12
|
+
const dir = join(process.cwd(), '.scale', 'qa', 'site-skills');
|
|
13
|
+
const hash = Buffer.from(domain).toString('hex').slice(0, 16);
|
|
14
|
+
const path = join(dir, `${hash}.json`);
|
|
15
|
+
if (!existsSync(path))
|
|
16
|
+
return null;
|
|
17
|
+
return JSON.parse(readFileSync(path, 'utf-8'));
|
|
18
|
+
}
|
|
19
|
+
//# sourceMappingURL=E2ETestOrchestrator.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"E2ETestOrchestrator.js","sourceRoot":"","sources":["../../src/qa/E2ETestOrchestrator.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,aAAa,EAAE,YAAY,EAAE,MAAM,SAAS,CAAA;AAC5E,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAA;AAShC,MAAM,UAAU,eAAe,CAAC,MAAc,EAAE,SAAiC,EAAE,KAAgD;IACjI,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,aAAa,CAAC,CAAA;IAC9D,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;QAAE,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAA;IACzD,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAA;IAC7D,MAAM,KAAK,GAAgB,EAAE,MAAM,EAAE,SAAS,EAAE,KAAK,EAAE,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAA;IAC1F,aAAa,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,OAAO,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAA;AAC1E,CAAC;AAED,MAAM,UAAU,eAAe,CAAC,MAAc;IAC5C,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,aAAa,CAAC,CAAA;IAC9D,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAA;IAC7D,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,OAAO,CAAC,CAAA;IACtC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;QAAE,OAAO,IAAI,CAAA;IAClC,OAAO,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,CAAA;AAChD,CAAC"}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
export interface ReviewFinding {
|
|
2
|
+
severity: 'critical' | 'high' | 'medium' | 'low';
|
|
3
|
+
category: string;
|
|
4
|
+
description: string;
|
|
5
|
+
suggestion?: string;
|
|
6
|
+
file?: string;
|
|
7
|
+
line?: number;
|
|
8
|
+
}
|
|
9
|
+
export interface ModelReview {
|
|
10
|
+
model: string;
|
|
11
|
+
findings: ReviewFinding[];
|
|
12
|
+
summary: string;
|
|
13
|
+
confidence: number;
|
|
14
|
+
}
|
|
15
|
+
export interface CrossModelReviewResult {
|
|
16
|
+
reviews: ModelReview[];
|
|
17
|
+
consensus: {
|
|
18
|
+
unanimous: ReviewFinding[];
|
|
19
|
+
majority: ReviewFinding[];
|
|
20
|
+
solo: ReviewFinding[];
|
|
21
|
+
};
|
|
22
|
+
overallScore: number;
|
|
23
|
+
recommendation: 'approve' | 'approve_with_comments' | 'needs_work' | 'blocked';
|
|
24
|
+
}
|
|
25
|
+
export declare class CrossModelReviewer {
|
|
26
|
+
/**
|
|
27
|
+
* Aggregate reviews from multiple models into a consensus.
|
|
28
|
+
* Each review contains findings — this merges and de-duplicates them.
|
|
29
|
+
*/
|
|
30
|
+
aggregate(reviews: ModelReview[]): CrossModelReviewResult;
|
|
31
|
+
/**
|
|
32
|
+
* Generate a human-readable consensus report.
|
|
33
|
+
*/
|
|
34
|
+
renderReport(result: CrossModelReviewResult): string;
|
|
35
|
+
}
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
export class CrossModelReviewer {
|
|
2
|
+
/**
|
|
3
|
+
* Aggregate reviews from multiple models into a consensus.
|
|
4
|
+
* Each review contains findings — this merges and de-duplicates them.
|
|
5
|
+
*/
|
|
6
|
+
aggregate(reviews) {
|
|
7
|
+
const allFindings = new Map();
|
|
8
|
+
for (const review of reviews) {
|
|
9
|
+
for (const finding of review.findings) {
|
|
10
|
+
// Key: category + description summary for dedup
|
|
11
|
+
const key = `${finding.category}:${(finding.description ?? '').slice(0, 80)}`;
|
|
12
|
+
const existing = allFindings.get(key);
|
|
13
|
+
if (existing) {
|
|
14
|
+
existing.models.push(review.model);
|
|
15
|
+
// Keep the more detailed version
|
|
16
|
+
if ((finding.suggestion?.length ?? 0) > (existing.finding.suggestion?.length ?? 0)) {
|
|
17
|
+
existing.finding = finding;
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
else {
|
|
21
|
+
allFindings.set(key, { finding, models: [review.model] });
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
const modelCount = reviews.length;
|
|
26
|
+
const majority = Math.ceil(modelCount / 2);
|
|
27
|
+
const unanimous = [];
|
|
28
|
+
const majorityList = [];
|
|
29
|
+
const solo = [];
|
|
30
|
+
for (const [, item] of allFindings) {
|
|
31
|
+
if (item.models.length === modelCount)
|
|
32
|
+
unanimous.push(item.finding);
|
|
33
|
+
else if (item.models.length >= majority)
|
|
34
|
+
majorityList.push(item.finding);
|
|
35
|
+
else
|
|
36
|
+
solo.push(item.finding);
|
|
37
|
+
}
|
|
38
|
+
// Score and recommendation
|
|
39
|
+
const criticalCount = [...allFindings.values()].filter(f => f.finding.severity === 'critical').length;
|
|
40
|
+
const highCount = [...allFindings.values()].filter(f => f.finding.severity === 'high').length;
|
|
41
|
+
const score = Math.max(0, 100 - (criticalCount * 25) - (highCount * 10) - (solo.length * 2));
|
|
42
|
+
let recommendation;
|
|
43
|
+
if (criticalCount > 0 && unanimous.some(f => f.severity === 'critical'))
|
|
44
|
+
recommendation = 'blocked';
|
|
45
|
+
else if (criticalCount > 0)
|
|
46
|
+
recommendation = 'needs_work';
|
|
47
|
+
else if (highCount > 0)
|
|
48
|
+
recommendation = 'approve_with_comments';
|
|
49
|
+
else
|
|
50
|
+
recommendation = 'approve';
|
|
51
|
+
return { reviews, consensus: { unanimous, majority: majorityList, solo }, overallScore: score, recommendation };
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Generate a human-readable consensus report.
|
|
55
|
+
*/
|
|
56
|
+
renderReport(result) {
|
|
57
|
+
const lines = [
|
|
58
|
+
`Cross-Model Review Report`,
|
|
59
|
+
`Models: ${result.reviews.map(r => r.model).join(', ')}`,
|
|
60
|
+
`Overall Score: ${result.overallScore}/100`,
|
|
61
|
+
`Recommendation: ${result.recommendation.toUpperCase()}`,
|
|
62
|
+
``,
|
|
63
|
+
`## Unanimous Findings (${result.consensus.unanimous.length})`,
|
|
64
|
+
...result.consensus.unanimous.map(f => ` [${f.severity.toUpperCase()}] ${f.category}: ${f.description}`),
|
|
65
|
+
``,
|
|
66
|
+
`## Majority Findings (${result.consensus.majority.length})`,
|
|
67
|
+
...result.consensus.majority.map(f => ` [${f.severity.toUpperCase()}] ${f.category}: ${f.description}`),
|
|
68
|
+
``,
|
|
69
|
+
`## Solo Findings (${result.consensus.solo.length})`,
|
|
70
|
+
...result.consensus.solo.map(f => ` [${f.severity.toUpperCase()}] ${f.category}: ${f.description} (${f.suggestion ?? 'no suggestion'})`),
|
|
71
|
+
];
|
|
72
|
+
return lines.join('\n');
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
//# sourceMappingURL=CrossModelReviewer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"CrossModelReviewer.js","sourceRoot":"","sources":["../../src/review/CrossModelReviewer.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { CrossModelReviewResult } from './CrossModelReviewer.js';
|
|
2
|
+
export interface ReviewSummary {
|
|
3
|
+
totalFindings: number;
|
|
4
|
+
bySeverity: Record<string, number>;
|
|
5
|
+
byCategory: Record<string, number>;
|
|
6
|
+
modelAgreement: number;
|
|
7
|
+
topIssues: Array<{
|
|
8
|
+
description: string;
|
|
9
|
+
severity: string;
|
|
10
|
+
models: string[];
|
|
11
|
+
}>;
|
|
12
|
+
}
|
|
13
|
+
export declare function summarizeReviews(result: CrossModelReviewResult): ReviewSummary;
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
export function summarizeReviews(result) {
|
|
2
|
+
const allFindings = result.reviews.flatMap(r => r.findings);
|
|
3
|
+
const bySeverity = {};
|
|
4
|
+
const byCategory = {};
|
|
5
|
+
for (const f of allFindings) {
|
|
6
|
+
bySeverity[f.severity] = (bySeverity[f.severity] ?? 0) + 1;
|
|
7
|
+
byCategory[f.category] = (byCategory[f.category] ?? 0) + 1;
|
|
8
|
+
}
|
|
9
|
+
const total = allFindings.length;
|
|
10
|
+
const consensusTotal = result.consensus.unanimous.length + result.consensus.majority.length;
|
|
11
|
+
const agreement = total > 0 ? consensusTotal / (consensusTotal + result.consensus.solo.length) : 1;
|
|
12
|
+
return {
|
|
13
|
+
totalFindings: total,
|
|
14
|
+
bySeverity,
|
|
15
|
+
byCategory,
|
|
16
|
+
modelAgreement: agreement,
|
|
17
|
+
topIssues: [...result.consensus.unanimous, ...result.consensus.majority]
|
|
18
|
+
.slice(0, 5)
|
|
19
|
+
.map(f => ({
|
|
20
|
+
description: f.description,
|
|
21
|
+
severity: f.severity,
|
|
22
|
+
models: result.reviews
|
|
23
|
+
.filter(r => r.findings.some(rf => rf.description === f.description))
|
|
24
|
+
.map(r => r.model),
|
|
25
|
+
})),
|
|
26
|
+
};
|
|
27
|
+
}
|
|
28
|
+
//# sourceMappingURL=ReviewAggregator.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ReviewAggregator.js","sourceRoot":"","sources":["../../src/review/ReviewAggregator.ts"],"names":[],"mappings":"AAUA,MAAM,UAAU,gBAAgB,CAAC,MAA8B;IAC7D,MAAM,WAAW,GAAG,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAA;IAC3D,MAAM,UAAU,GAA2B,EAAE,CAAA;IAC7C,MAAM,UAAU,GAA2B,EAAE,CAAA;IAE7C,KAAK,MAAM,CAAC,IAAI,WAAW,EAAE,CAAC;QAC5B,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAA;QAC1D,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAA;IAC5D,CAAC;IAED,MAAM,KAAK,GAAG,WAAW,CAAC,MAAM,CAAA;IAChC,MAAM,cAAc,GAAG,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,MAAM,GAAG,MAAM,CAAC,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAA;IAC3F,MAAM,SAAS,GAAG,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,CAAC,cAAc,GAAG,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAA;IAElG,OAAO;QACL,aAAa,EAAE,KAAK;QACpB,UAAU;QACV,UAAU;QACV,cAAc,EAAE,SAAS;QACzB,SAAS,EAAE,CAAC,GAAG,MAAM,CAAC,SAAS,CAAC,SAAS,EAAE,GAAG,MAAM,CAAC,SAAS,CAAC,QAAQ,CAAC;aACrE,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC;aACX,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YACT,WAAW,EAAE,CAAC,CAAC,WAAW;YAC1B,QAAQ,EAAE,CAAC,CAAC,QAAQ;YACpB,MAAM,EAAE,MAAM,CAAC,OAAO;iBACnB,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,WAAW,KAAK,CAAC,CAAC,WAAW,CAAC,CAAC;iBACpE,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC;SACrB,CAAC,CAAC;KACN,CAAA;AACH,CAAC"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
export declare const reviewCommand: import("citty").CommandDef<{
|
|
2
|
+
diff: {
|
|
3
|
+
type: "string";
|
|
4
|
+
description: string;
|
|
5
|
+
};
|
|
6
|
+
models: {
|
|
7
|
+
type: "string";
|
|
8
|
+
default: string;
|
|
9
|
+
description: string;
|
|
10
|
+
};
|
|
11
|
+
files: {
|
|
12
|
+
type: "string";
|
|
13
|
+
description: string;
|
|
14
|
+
};
|
|
15
|
+
}>;
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { defineCommand } from 'citty';
|
|
2
|
+
export const reviewCommand = defineCommand({
|
|
3
|
+
meta: {
|
|
4
|
+
name: 'review',
|
|
5
|
+
description: 'Cross-model review — multiple models independently review a change',
|
|
6
|
+
},
|
|
7
|
+
args: {
|
|
8
|
+
diff: { type: 'string', description: 'Git diff reference (e.g. HEAD~1)' },
|
|
9
|
+
models: { type: 'string', default: 'claude-sonnet', description: 'Comma-separated model list' },
|
|
10
|
+
files: { type: 'string', description: 'Specific files to review' },
|
|
11
|
+
},
|
|
12
|
+
async run({ args }) {
|
|
13
|
+
const { CrossModelReviewer } = await import('./CrossModelReviewer.js');
|
|
14
|
+
const reviewer = new CrossModelReviewer();
|
|
15
|
+
console.log('Cross-model review requested');
|
|
16
|
+
console.log(` Models: ${args.models}`);
|
|
17
|
+
console.log(` Diff: ${args.diff ?? 'HEAD'}`);
|
|
18
|
+
console.log('\nNote: Actual model invocation is delegated to the hosting agent.\n');
|
|
19
|
+
console.log('Review framework initialized. Use individual model outputs as input to:');
|
|
20
|
+
console.log(' reviewer.aggregate([review1, review2, ...])');
|
|
21
|
+
console.log(' reviewer.renderReport(result)');
|
|
22
|
+
},
|
|
23
|
+
});
|
|
24
|
+
//# sourceMappingURL=reviewCommands.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"reviewCommands.js","sourceRoot":"","sources":["../../src/review/reviewCommands.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,OAAO,CAAA;AAErC,MAAM,CAAC,MAAM,aAAa,GAAG,aAAa,CAAC;IACzC,IAAI,EAAE;QACJ,IAAI,EAAE,QAAQ;QACd,WAAW,EAAE,oEAAoE;KAClF;IACD,IAAI,EAAE;QACJ,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,WAAW,EAAE,kCAAkC,EAAE;QACzE,MAAM,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,eAAe,EAAE,WAAW,EAAE,4BAA4B,EAAE;QAC/F,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,WAAW,EAAE,0BAA0B,EAAE;KACnE;IACD,KAAK,CAAC,GAAG,CAAC,EAAE,IAAI,EAAE;QAChB,MAAM,EAAE,kBAAkB,EAAE,GAAG,MAAM,MAAM,CAAC,yBAAyB,CAAC,CAAA;QACtE,MAAM,QAAQ,GAAG,IAAI,kBAAkB,EAAE,CAAA;QAEzC,OAAO,CAAC,GAAG,CAAC,8BAA8B,CAAC,CAAA;QAC3C,OAAO,CAAC,GAAG,CAAC,aAAa,IAAI,CAAC,MAAM,EAAE,CAAC,CAAA;QACvC,OAAO,CAAC,GAAG,CAAC,WAAW,IAAI,CAAC,IAAI,IAAI,MAAM,EAAE,CAAC,CAAA;QAC7C,OAAO,CAAC,GAAG,CAAC,sEAAsE,CAAC,CAAA;QACnF,OAAO,CAAC,GAAG,CAAC,yEAAyE,CAAC,CAAA;QACtF,OAAO,CAAC,GAAG,CAAC,+CAA+C,CAAC,CAAA;QAC5D,OAAO,CAAC,GAAG,CAAC,iCAAiC,CAAC,CAAA;IAChD,CAAC;CACF,CAAC,CAAA"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
export interface LocalModelConfig {
|
|
2
|
+
name: string;
|
|
3
|
+
baseUrl: string;
|
|
4
|
+
apiKey: string;
|
|
5
|
+
maxTokens: number;
|
|
6
|
+
}
|
|
7
|
+
export declare function resolveLocalModelConfig(): LocalModelConfig;
|
|
8
|
+
export declare function getSupportedLocalModels(): Array<{
|
|
9
|
+
id: string;
|
|
10
|
+
description: string;
|
|
11
|
+
}>;
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
export function resolveLocalModelConfig() {
|
|
2
|
+
const name = process.env.SCALE_LOCAL_MODEL ?? 'qwen-2.5-72b';
|
|
3
|
+
const baseUrl = process.env.SCALE_LOCAL_BASE_URL ?? process.env.OPENAI_BASE_URL ?? 'http://localhost:11434/v1';
|
|
4
|
+
const apiKey = process.env.SCALE_LOCAL_API_KEY ?? process.env.OPENAI_API_KEY ?? 'ollama';
|
|
5
|
+
return {
|
|
6
|
+
name,
|
|
7
|
+
baseUrl: baseUrl.replace(/\/+$/, ''),
|
|
8
|
+
apiKey,
|
|
9
|
+
maxTokens: parseInt(process.env.SCALE_LOCAL_MAX_TOKENS ?? '32000', 10),
|
|
10
|
+
};
|
|
11
|
+
}
|
|
12
|
+
export function getSupportedLocalModels() {
|
|
13
|
+
return [
|
|
14
|
+
{ id: 'qwen-2.5-72b', description: 'Qwen 2.5 72B — Alibaba Cloud / local deployment' },
|
|
15
|
+
{ id: 'glm-4-plus', description: 'GLM-4 Plus — Zhipu AI / local deployment' },
|
|
16
|
+
{ id: 'deepseek-v3', description: 'DeepSeek V3 — DeepSeek / local deployment' },
|
|
17
|
+
{ id: 'llama-3.1-70b', description: 'Llama 3.1 70B — Meta / Ollama' },
|
|
18
|
+
{ id: 'qwen-2.5-7b', description: 'Qwen 2.5 7B — lightweight, fits consumer GPU' },
|
|
19
|
+
];
|
|
20
|
+
}
|
|
21
|
+
//# sourceMappingURL=LocalModelProvider.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LocalModelProvider.js","sourceRoot":"","sources":["../../src/routing/LocalModelProvider.ts"],"names":[],"mappings":"AAOA,MAAM,UAAU,uBAAuB;IACrC,MAAM,IAAI,GAAG,OAAO,CAAC,GAAG,CAAC,iBAAiB,IAAI,cAAc,CAAA;IAC5D,MAAM,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,oBAAoB,IAAI,OAAO,CAAC,GAAG,CAAC,eAAe,IAAI,2BAA2B,CAAA;IAC9G,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,mBAAmB,IAAI,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,QAAQ,CAAA;IAExF,OAAO;QACL,IAAI;QACJ,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC;QACpC,MAAM;QACN,SAAS,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,sBAAsB,IAAI,OAAO,EAAE,EAAE,CAAC;KACvE,CAAA;AACH,CAAC;AAED,MAAM,UAAU,uBAAuB;IACrC,OAAO;QACL,EAAE,EAAE,EAAE,cAAc,EAAE,WAAW,EAAE,iDAAiD,EAAE;QACtF,EAAE,EAAE,EAAE,YAAY,EAAE,WAAW,EAAE,0CAA0C,EAAE;QAC7E,EAAE,EAAE,EAAE,aAAa,EAAE,WAAW,EAAE,2CAA2C,EAAE;QAC/E,EAAE,EAAE,EAAE,eAAe,EAAE,WAAW,EAAE,+BAA+B,EAAE;QACrE,EAAE,EAAE,EAAE,aAAa,EAAE,WAAW,EAAE,8CAA8C,EAAE;KACnF,CAAA;AACH,CAAC"}
|
|
@@ -5,19 +5,29 @@ export interface ModelConfig {
|
|
|
5
5
|
name: string;
|
|
6
6
|
maxTokens: number;
|
|
7
7
|
costPerMToken: number;
|
|
8
|
+
modalities: string[];
|
|
8
9
|
}
|
|
9
10
|
export declare const DEFAULT_MODELS: Record<ModelTier, ModelConfig>;
|
|
11
|
+
export declare const LOCAL_MODELS: Record<string, ModelConfig>;
|
|
10
12
|
export interface RoutingContext {
|
|
11
13
|
taskComplexity?: number;
|
|
12
14
|
artifactType?: string;
|
|
13
15
|
stepCount?: number;
|
|
14
16
|
previousFailures?: number;
|
|
15
17
|
budget?: 'low' | 'medium' | 'high';
|
|
18
|
+
modality?: 'text' | 'vision';
|
|
19
|
+
}
|
|
20
|
+
export interface RouterConfig {
|
|
21
|
+
baseUrl?: string;
|
|
22
|
+
apiKey?: string;
|
|
23
|
+
localModelName?: string;
|
|
16
24
|
}
|
|
17
25
|
export interface IModelRouter {
|
|
18
26
|
route(ctx: RoutingContext): ModelConfig;
|
|
19
27
|
getModels(): Record<ModelTier, ModelConfig>;
|
|
20
28
|
setModel(tier: ModelTier, config: ModelConfig): void;
|
|
29
|
+
preCheck(ctx: RoutingContext): ModelConfig;
|
|
30
|
+
setLocalModel(name: string, config: Partial<ModelConfig>): void;
|
|
21
31
|
}
|
|
22
32
|
export declare class ModelRouter implements IModelRouter {
|
|
23
33
|
private eventBus;
|
|
@@ -26,5 +36,7 @@ export declare class ModelRouter implements IModelRouter {
|
|
|
26
36
|
route(ctx: RoutingContext): ModelConfig;
|
|
27
37
|
getModels(): Record<ModelTier, ModelConfig>;
|
|
28
38
|
setModel(tier: ModelTier, config: ModelConfig): void;
|
|
39
|
+
preCheck(ctx: RoutingContext): ModelConfig;
|
|
40
|
+
setLocalModel(name: string, config: Partial<ModelConfig>): void;
|
|
29
41
|
private explainRouting;
|
|
30
42
|
}
|
|
@@ -3,10 +3,16 @@
|
|
|
3
3
|
// 设计参考:docs/01-ARCHITECTURE.md §二 L4
|
|
4
4
|
import { logger } from '../core/logger.js';
|
|
5
5
|
export const DEFAULT_MODELS = {
|
|
6
|
-
fast: { tier: 'fast', name: 'claude-haiku', maxTokens: 200_000, costPerMToken: 0.25 },
|
|
7
|
-
balanced: { tier: 'balanced', name: 'claude-sonnet', maxTokens: 200_000, costPerMToken: 3.0 },
|
|
8
|
-
powerful: { tier: 'powerful', name: 'claude-opus', maxTokens: 200_000, costPerMToken: 15.0 },
|
|
9
|
-
local: { tier: 'local', name: 'local-llm', maxTokens: 32_000, costPerMToken: 0.0 },
|
|
6
|
+
fast: { tier: 'fast', name: 'claude-haiku', maxTokens: 200_000, costPerMToken: 0.25, modalities: ['text'] },
|
|
7
|
+
balanced: { tier: 'balanced', name: 'claude-sonnet', maxTokens: 200_000, costPerMToken: 3.0, modalities: ['text', 'vision'] },
|
|
8
|
+
powerful: { tier: 'powerful', name: 'claude-opus', maxTokens: 200_000, costPerMToken: 15.0, modalities: ['text', 'vision'] },
|
|
9
|
+
local: { tier: 'local', name: 'local-llm', maxTokens: 32_000, costPerMToken: 0.0, modalities: ['text'] },
|
|
10
|
+
};
|
|
11
|
+
// Local model registry for China market
|
|
12
|
+
export const LOCAL_MODELS = {
|
|
13
|
+
'qwen-2.5-72b': { tier: 'local', name: 'qwen-2.5-72b', maxTokens: 32_000, costPerMToken: 0.0, modalities: ['text'] },
|
|
14
|
+
'glm-4-plus': { tier: 'local', name: 'glm-4-plus', maxTokens: 128_000, costPerMToken: 0.0, modalities: ['text', 'vision'] },
|
|
15
|
+
'deepseek-v3': { tier: 'local', name: 'deepseek-v3', maxTokens: 64_000, costPerMToken: 0.0, modalities: ['text'] },
|
|
10
16
|
};
|
|
11
17
|
export class ModelRouter {
|
|
12
18
|
constructor(eventBus, models) {
|
|
@@ -34,6 +40,14 @@ export class ModelRouter {
|
|
|
34
40
|
else {
|
|
35
41
|
tier = 'balanced';
|
|
36
42
|
}
|
|
43
|
+
// If vision task and selected tier lacks vision, upgrade
|
|
44
|
+
if (ctx.modality === 'vision' && !(this.models[tier].modalities ?? ['text']).includes('vision')) {
|
|
45
|
+
// Upgrade to next tier that supports vision
|
|
46
|
+
if ((this.models.balanced.modalities ?? ['text']).includes('vision'))
|
|
47
|
+
tier = 'balanced';
|
|
48
|
+
else if ((this.models.powerful.modalities ?? ['text']).includes('vision'))
|
|
49
|
+
tier = 'powerful';
|
|
50
|
+
}
|
|
37
51
|
const model = this.models[tier];
|
|
38
52
|
logger.debug({ ctx, selectedTier: tier, model: model.name }, 'Model routed');
|
|
39
53
|
this.eventBus.emit('tool.called', {
|
|
@@ -50,6 +64,19 @@ export class ModelRouter {
|
|
|
50
64
|
setModel(tier, config) {
|
|
51
65
|
this.models[tier] = config;
|
|
52
66
|
}
|
|
67
|
+
preCheck(ctx) {
|
|
68
|
+
// For pre-check gates, always try local model first (cheaper)
|
|
69
|
+
return this.models.local;
|
|
70
|
+
}
|
|
71
|
+
setLocalModel(name, config) {
|
|
72
|
+
const localModel = LOCAL_MODELS[name];
|
|
73
|
+
if (localModel) {
|
|
74
|
+
this.models.local = { ...localModel, ...config };
|
|
75
|
+
}
|
|
76
|
+
else {
|
|
77
|
+
this.models.local = { tier: 'local', name, maxTokens: 32_000, costPerMToken: 0.0, modalities: ['text'], ...config };
|
|
78
|
+
}
|
|
79
|
+
}
|
|
53
80
|
explainRouting(ctx, tier) {
|
|
54
81
|
if (ctx.budget === 'low')
|
|
55
82
|
return 'budget=low → fast';
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ModelRouter.js","sourceRoot":"","sources":["../../src/routing/ModelRouter.ts"],"names":[],"mappings":"AAAA,mCAAmC;AACnC,cAAc;AACd,qCAAqC;AAGrC,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAA;
|
|
1
|
+
{"version":3,"file":"ModelRouter.js","sourceRoot":"","sources":["../../src/routing/ModelRouter.ts"],"names":[],"mappings":"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"}
|
|
@@ -482,3 +482,4 @@ export declare function createAiOsMigration(input?: AiOsMigrationInput): AiOsMig
|
|
|
482
482
|
export declare function createAiOsDoctor(input?: AiOsDoctorInput): AiOsDoctorReport;
|
|
483
483
|
export declare function createAiOsAdoption(input: AiOsAdoptionInput): Promise<AiOsAdoptionReport>;
|
|
484
484
|
export declare function createAiOsStatus(input?: AiOsStatusInput): AiOsStatusReport;
|
|
485
|
+
export declare function buildCostRoiEstimate(projectDir?: string): Promise<string>;
|
|
@@ -1822,6 +1822,21 @@ function recommendations(options) {
|
|
|
1822
1822
|
}
|
|
1823
1823
|
return output;
|
|
1824
1824
|
}
|
|
1825
|
+
export async function buildCostRoiEstimate(projectDir) {
|
|
1826
|
+
const { CostAnalyzer } = await import('./CostAnalyzer.js');
|
|
1827
|
+
const analyzer = new CostAnalyzer();
|
|
1828
|
+
const records = analyzer.loadRecords();
|
|
1829
|
+
const breakdown = analyzer.analyze(records);
|
|
1830
|
+
const suggestions = analyzer.suggestOptimizations(breakdown);
|
|
1831
|
+
const lines = [
|
|
1832
|
+
'\n--- ROI Estimate ---',
|
|
1833
|
+
` Governance cost (est.): $${breakdown.total.cost.toFixed(2)}/month`,
|
|
1834
|
+
` Estimated defect prevention value: $${(breakdown.total.cost * 50).toFixed(2)}/month`,
|
|
1835
|
+
` Net estimated ROI: ${((breakdown.total.cost * 50 - breakdown.total.cost) / Math.max(0.01, breakdown.total.cost)).toFixed(0)}x`,
|
|
1836
|
+
` Potential savings: $${suggestions.reduce((s, sug) => s + sug.estimatedMonthlySavings, 0).toFixed(2)}/month`,
|
|
1837
|
+
];
|
|
1838
|
+
return lines.join('\n');
|
|
1839
|
+
}
|
|
1825
1840
|
function normalizeSkillTaskLevel(value) {
|
|
1826
1841
|
const normalized = String(value ?? 'M').trim().toUpperCase();
|
|
1827
1842
|
if (normalized === 'S' || normalized === 'M' || normalized === 'L' || normalized === 'CRITICAL')
|