@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.
Files changed (227) hide show
  1. package/README.md +30 -2
  2. package/dist/api/cli.js +43 -2
  3. package/dist/api/cli.js.map +1 -1
  4. package/dist/api/quickstart.d.ts +11 -0
  5. package/dist/api/quickstart.js +98 -1
  6. package/dist/api/quickstart.js.map +1 -1
  7. package/dist/artifact/fsmDefinitions.js +15 -2
  8. package/dist/artifact/fsmDefinitions.js.map +1 -1
  9. package/dist/artifact/types.d.ts +1 -1
  10. package/dist/artifact/types.js.map +1 -1
  11. package/dist/cache/ScanCache.d.ts +41 -0
  12. package/dist/cache/ScanCache.js +120 -0
  13. package/dist/cache/ScanCache.js.map +1 -0
  14. package/dist/capabilities/BrowserQACapability.d.ts +14 -0
  15. package/dist/capabilities/BrowserQACapability.js +94 -0
  16. package/dist/capabilities/BrowserQACapability.js.map +1 -1
  17. package/dist/cli/autofixCommands.d.ts +22 -0
  18. package/dist/cli/autofixCommands.js +32 -0
  19. package/dist/cli/autofixCommands.js.map +1 -0
  20. package/dist/cli/cortexCommands.d.ts +71 -0
  21. package/dist/cli/cortexCommands.js +335 -0
  22. package/dist/cli/cortexCommands.js.map +1 -0
  23. package/dist/cli/costCommands.d.ts +13 -0
  24. package/dist/cli/costCommands.js +48 -0
  25. package/dist/cli/costCommands.js.map +1 -0
  26. package/dist/cli/orchCommands.d.ts +43 -0
  27. package/dist/cli/orchCommands.js +135 -0
  28. package/dist/cli/orchCommands.js.map +1 -0
  29. package/dist/cli/phaseCommands.js +1 -2
  30. package/dist/cli/phaseCommands.js.map +1 -1
  31. package/dist/cli/qaCommands.d.ts +22 -0
  32. package/dist/cli/qaCommands.js +84 -0
  33. package/dist/cli/qaCommands.js.map +1 -0
  34. package/dist/cli/quickstartCommands.d.ts +17 -0
  35. package/dist/cli/quickstartCommands.js +47 -0
  36. package/dist/cli/quickstartCommands.js.map +1 -0
  37. package/dist/cli/shieldCommands.d.ts +30 -0
  38. package/dist/cli/shieldCommands.js +212 -0
  39. package/dist/cli/shieldCommands.js.map +1 -0
  40. package/dist/cli/tuiCommands.d.ts +7 -0
  41. package/dist/cli/tuiCommands.js +33 -0
  42. package/dist/cli/tuiCommands.js.map +1 -0
  43. package/dist/codegraph/CodeIntelligence.d.ts +27 -0
  44. package/dist/codegraph/CodeIntelligence.js +316 -3
  45. package/dist/codegraph/CodeIntelligence.js.map +1 -1
  46. package/dist/config/profiles.js +26 -0
  47. package/dist/config/profiles.js.map +1 -1
  48. package/dist/cortex/GovernanceMetrics.d.ts +66 -0
  49. package/dist/cortex/GovernanceMetrics.js +230 -0
  50. package/dist/cortex/GovernanceMetrics.js.map +1 -0
  51. package/dist/cortex/InstinctExtractor.d.ts +61 -0
  52. package/dist/cortex/InstinctExtractor.js +184 -0
  53. package/dist/cortex/InstinctExtractor.js.map +1 -0
  54. package/dist/cortex/InstinctStore.d.ts +54 -0
  55. package/dist/cortex/InstinctStore.js +266 -0
  56. package/dist/cortex/InstinctStore.js.map +1 -0
  57. package/dist/cortex/ReflexionEngine.d.ts +34 -0
  58. package/dist/cortex/ReflexionEngine.js +157 -0
  59. package/dist/cortex/ReflexionEngine.js.map +1 -0
  60. package/dist/cortex/SessionInjector.d.ts +44 -0
  61. package/dist/cortex/SessionInjector.js +127 -0
  62. package/dist/cortex/SessionInjector.js.map +1 -0
  63. package/dist/cortex/adapters/ClaudeAdapter.d.ts +17 -0
  64. package/dist/cortex/adapters/ClaudeAdapter.js +61 -0
  65. package/dist/cortex/adapters/ClaudeAdapter.js.map +1 -0
  66. package/dist/cortex/adapters/CodexAdapter.d.ts +10 -0
  67. package/dist/cortex/adapters/CodexAdapter.js +52 -0
  68. package/dist/cortex/adapters/CodexAdapter.js.map +1 -0
  69. package/dist/cortex/adapters/CursorAdapter.d.ts +10 -0
  70. package/dist/cortex/adapters/CursorAdapter.js +46 -0
  71. package/dist/cortex/adapters/CursorAdapter.js.map +1 -0
  72. package/dist/cortex/adapters/GeminiAdapter.d.ts +11 -0
  73. package/dist/cortex/adapters/GeminiAdapter.js +48 -0
  74. package/dist/cortex/adapters/GeminiAdapter.js.map +1 -0
  75. package/dist/dashboard/DashboardServer.d.ts +33 -13
  76. package/dist/dashboard/DashboardServer.js +314 -182
  77. package/dist/dashboard/DashboardServer.js.map +1 -1
  78. package/dist/dashboard/index.d.ts +2 -2
  79. package/dist/dashboard/index.js +1 -1
  80. package/dist/dashboard/index.js.map +1 -1
  81. package/dist/dashboard/server.d.ts +8 -22
  82. package/dist/dashboard/server.js +2 -83
  83. package/dist/dashboard/server.js.map +1 -1
  84. package/dist/eval/BenchmarkPublisher.d.ts +25 -0
  85. package/dist/eval/BenchmarkPublisher.js +27 -0
  86. package/dist/eval/BenchmarkPublisher.js.map +1 -0
  87. package/dist/guardrails/DependencyAuditor.js +10 -1
  88. package/dist/guardrails/DependencyAuditor.js.map +1 -1
  89. package/dist/index.d.ts +1 -1
  90. package/dist/index.js +1 -1
  91. package/dist/index.js.map +1 -1
  92. package/dist/orchestrator/OrchestratorDaemon.d.ts +44 -0
  93. package/dist/orchestrator/OrchestratorDaemon.js +150 -0
  94. package/dist/orchestrator/OrchestratorDaemon.js.map +1 -0
  95. package/dist/orchestrator/PolicyLoader.d.ts +80 -0
  96. package/dist/orchestrator/PolicyLoader.js +229 -0
  97. package/dist/orchestrator/PolicyLoader.js.map +1 -0
  98. package/dist/orchestrator/ReconciliationLoop.d.ts +71 -0
  99. package/dist/orchestrator/ReconciliationLoop.js +266 -0
  100. package/dist/orchestrator/ReconciliationLoop.js.map +1 -0
  101. package/dist/orchestrator/TrackerAdapter.d.ts +60 -0
  102. package/dist/orchestrator/TrackerAdapter.js +147 -0
  103. package/dist/orchestrator/TrackerAdapter.js.map +1 -0
  104. package/dist/orchestrator/WorkspaceManager.d.ts +66 -0
  105. package/dist/orchestrator/WorkspaceManager.js +257 -0
  106. package/dist/orchestrator/WorkspaceManager.js.map +1 -0
  107. package/dist/qa/BrowserDaemon.d.ts +23 -0
  108. package/dist/qa/BrowserDaemon.js +79 -0
  109. package/dist/qa/BrowserDaemon.js.map +1 -0
  110. package/dist/qa/E2ETestOrchestrator.d.ts +14 -0
  111. package/dist/qa/E2ETestOrchestrator.js +19 -0
  112. package/dist/qa/E2ETestOrchestrator.js.map +1 -0
  113. package/dist/review/CrossModelReviewer.d.ts +35 -0
  114. package/dist/review/CrossModelReviewer.js +75 -0
  115. package/dist/review/CrossModelReviewer.js.map +1 -0
  116. package/dist/review/ReviewAggregator.d.ts +13 -0
  117. package/dist/review/ReviewAggregator.js +28 -0
  118. package/dist/review/ReviewAggregator.js.map +1 -0
  119. package/dist/review/reviewCommands.d.ts +15 -0
  120. package/dist/review/reviewCommands.js +24 -0
  121. package/dist/review/reviewCommands.js.map +1 -0
  122. package/dist/routing/LocalModelProvider.d.ts +11 -0
  123. package/dist/routing/LocalModelProvider.js +21 -0
  124. package/dist/routing/LocalModelProvider.js.map +1 -0
  125. package/dist/routing/ModelRouter.d.ts +12 -0
  126. package/dist/routing/ModelRouter.js +31 -4
  127. package/dist/routing/ModelRouter.js.map +1 -1
  128. package/dist/runtime/AiOsRuntime.d.ts +1 -0
  129. package/dist/runtime/AiOsRuntime.js +15 -0
  130. package/dist/runtime/AiOsRuntime.js.map +1 -1
  131. package/dist/runtime/CostAnalyzer.d.ts +53 -0
  132. package/dist/runtime/CostAnalyzer.js +160 -0
  133. package/dist/runtime/CostAnalyzer.js.map +1 -0
  134. package/dist/runtime/CostOptimizer.d.ts +11 -0
  135. package/dist/runtime/CostOptimizer.js +21 -0
  136. package/dist/runtime/CostOptimizer.js.map +1 -0
  137. package/dist/shield/PolicyCompiler.d.ts +70 -0
  138. package/dist/shield/PolicyCompiler.js +540 -0
  139. package/dist/shield/PolicyCompiler.js.map +1 -0
  140. package/dist/shield/ProtectedPaths.d.ts +39 -0
  141. package/dist/shield/ProtectedPaths.js +179 -0
  142. package/dist/shield/ProtectedPaths.js.map +1 -0
  143. package/dist/shield/ShieldProtocol.d.ts +50 -0
  144. package/dist/shield/ShieldProtocol.js +103 -0
  145. package/dist/shield/ShieldProtocol.js.map +1 -0
  146. package/dist/skills/SkillMdStandard.d.ts +33 -0
  147. package/dist/skills/SkillMdStandard.js +88 -0
  148. package/dist/skills/SkillMdStandard.js.map +1 -0
  149. package/dist/skills/SkillRegistry.d.ts +9 -1
  150. package/dist/skills/SkillRegistry.js +20 -0
  151. package/dist/skills/SkillRegistry.js.map +1 -1
  152. package/dist/skills/interop/GStackInterop.d.ts +15 -0
  153. package/dist/skills/interop/GStackInterop.js +34 -0
  154. package/dist/skills/interop/GStackInterop.js.map +1 -0
  155. package/dist/skills/interop/OMCInterop.d.ts +15 -0
  156. package/dist/skills/interop/OMCInterop.js +34 -0
  157. package/dist/skills/interop/OMCInterop.js.map +1 -0
  158. package/dist/topology/DomainMapper.d.ts +23 -0
  159. package/dist/topology/DomainMapper.js +179 -0
  160. package/dist/topology/DomainMapper.js.map +1 -0
  161. package/dist/topology/LayerClassifier.d.ts +8 -0
  162. package/dist/topology/LayerClassifier.js +109 -0
  163. package/dist/topology/LayerClassifier.js.map +1 -0
  164. package/dist/topology/TourGenerator.d.ts +18 -0
  165. package/dist/topology/TourGenerator.js +120 -0
  166. package/dist/topology/TourGenerator.js.map +1 -0
  167. package/dist/topology/index.d.ts +3 -0
  168. package/dist/topology/index.js +4 -0
  169. package/dist/topology/index.js.map +1 -0
  170. package/dist/tui/TuiDashboard.d.ts +3 -0
  171. package/dist/tui/TuiDashboard.js +120 -0
  172. package/dist/tui/TuiDashboard.js.map +1 -0
  173. package/dist/workflow/GateCatalog.d.ts +2 -0
  174. package/dist/workflow/GateCatalog.js +59 -3
  175. package/dist/workflow/GateCatalog.js.map +1 -1
  176. package/dist/workflow/GovernanceTemplatePacks.d.ts +1 -1
  177. package/dist/workflow/GovernanceTemplatePacks.js +15 -0
  178. package/dist/workflow/GovernanceTemplatePacks.js.map +1 -1
  179. package/dist/workflow/TddLoop.d.ts +2 -0
  180. package/dist/workflow/TddLoop.js +2 -0
  181. package/dist/workflow/TddLoop.js.map +1 -1
  182. package/dist/workflow/UpgradeManager.d.ts +10 -1
  183. package/dist/workflow/UpgradeManager.js +55 -0
  184. package/dist/workflow/UpgradeManager.js.map +1 -1
  185. package/dist/workflow/VerificationProfile.d.ts +8 -0
  186. package/dist/workflow/VerificationProfile.js +61 -0
  187. package/dist/workflow/VerificationProfile.js.map +1 -1
  188. package/dist/workflow/VerificationSchema.d.ts +46 -0
  189. package/dist/workflow/VerificationSchema.js +97 -0
  190. package/dist/workflow/VerificationSchema.js.map +1 -0
  191. package/dist/workflow/autofix/AutoFixEngine.d.ts +37 -0
  192. package/dist/workflow/autofix/AutoFixEngine.js +169 -0
  193. package/dist/workflow/autofix/AutoFixEngine.js.map +1 -0
  194. package/dist/workflow/execution/RalphEngine.d.ts +18 -0
  195. package/dist/workflow/execution/RalphEngine.js +22 -0
  196. package/dist/workflow/execution/RalphEngine.js.map +1 -1
  197. package/dist/workflow/gates/EnhancedGates.d.ts +74 -0
  198. package/dist/workflow/gates/EnhancedGates.js +653 -0
  199. package/dist/workflow/gates/EnhancedGates.js.map +1 -0
  200. package/dist/workflow/gates/GateSystem.d.ts +3 -0
  201. package/dist/workflow/gates/GateSystem.js +94 -1
  202. package/dist/workflow/gates/GateSystem.js.map +1 -1
  203. package/dist/workflow/types.d.ts +1 -1
  204. package/docs/README.md +3 -0
  205. package/docs/guides/DEVELOPMENT_WORKFLOW.md +28 -9
  206. package/docs/guides/GETTING_STARTED.md +19 -0
  207. package/docs/guides/MIGRATION.md +119 -0
  208. package/docs/workflow/GATES_AND_SCORE.md +34 -1
  209. package/docs/workflow/README.md +58 -10
  210. package/package.json +7 -17
  211. package/docs/ACTIVE_SECURITY_VISUAL_GATES.md +0 -87
  212. package/docs/AI_ENGINEERING_OS_POSITIONING.md +0 -607
  213. package/docs/BACKGROUND_HUNTER.md +0 -62
  214. package/docs/CODE_INTELLIGENCE.md +0 -180
  215. package/docs/CONTEXT_BUDGET.md +0 -165
  216. package/docs/DEPENDENCY_AUDIT.md +0 -118
  217. package/docs/EVOLUTION_SHADOW_MODE.md +0 -63
  218. package/docs/GITLAB_FLOW.md +0 -125
  219. package/docs/GOVERNANCE_DASHBOARD.md +0 -92
  220. package/docs/MEMORY_BRAIN.md +0 -104
  221. package/docs/MEMORY_FABRIC.md +0 -161
  222. package/docs/RESOURCE_GOVERNANCE.md +0 -92
  223. package/docs/RUNTIME_EVIDENCE.md +0 -101
  224. package/docs/WORKFLOW_EVAL.md +0 -151
  225. package/image/wechat-public.jpg +0 -0
  226. package/image/wxPay.jpg +0 -0
  227. 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":"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"}
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')