@hongmaple0820/scale-engine 0.40.1 → 0.43.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 (228) hide show
  1. package/README.md +30 -2
  2. package/dist/api/cli.js +286 -7
  3. package/dist/api/cli.js.map +1 -1
  4. package/dist/api/doctor.js +1 -1
  5. package/dist/api/doctor.js.map +1 -1
  6. package/dist/api/quickstart.d.ts +11 -0
  7. package/dist/api/quickstart.js +98 -1
  8. package/dist/api/quickstart.js.map +1 -1
  9. package/dist/artifact/fsmDefinitions.js +15 -2
  10. package/dist/artifact/fsmDefinitions.js.map +1 -1
  11. package/dist/artifact/types.d.ts +1 -1
  12. package/dist/artifact/types.js.map +1 -1
  13. package/dist/bootstrap/DependencyBootstrap.d.ts +1 -0
  14. package/dist/bootstrap/DependencyBootstrap.js +137 -25
  15. package/dist/bootstrap/DependencyBootstrap.js.map +1 -1
  16. package/dist/cache/ScanCache.d.ts +41 -0
  17. package/dist/cache/ScanCache.js +120 -0
  18. package/dist/cache/ScanCache.js.map +1 -0
  19. package/dist/capabilities/BrowserQACapability.d.ts +14 -0
  20. package/dist/capabilities/BrowserQACapability.js +94 -0
  21. package/dist/capabilities/BrowserQACapability.js.map +1 -1
  22. package/dist/capabilities/InstalledSkillsIntegration.js +29 -9
  23. package/dist/capabilities/InstalledSkillsIntegration.js.map +1 -1
  24. package/dist/cli/autofixCommands.d.ts +22 -0
  25. package/dist/cli/autofixCommands.js +32 -0
  26. package/dist/cli/autofixCommands.js.map +1 -0
  27. package/dist/cli/cortexCommands.d.ts +71 -0
  28. package/dist/cli/cortexCommands.js +335 -0
  29. package/dist/cli/cortexCommands.js.map +1 -0
  30. package/dist/cli/costCommands.d.ts +13 -0
  31. package/dist/cli/costCommands.js +48 -0
  32. package/dist/cli/costCommands.js.map +1 -0
  33. package/dist/cli/orchCommands.d.ts +43 -0
  34. package/dist/cli/orchCommands.js +135 -0
  35. package/dist/cli/orchCommands.js.map +1 -0
  36. package/dist/cli/phaseCommands.js +1 -2
  37. package/dist/cli/phaseCommands.js.map +1 -1
  38. package/dist/cli/qaCommands.d.ts +22 -0
  39. package/dist/cli/qaCommands.js +84 -0
  40. package/dist/cli/qaCommands.js.map +1 -0
  41. package/dist/cli/quickstartCommands.d.ts +17 -0
  42. package/dist/cli/quickstartCommands.js +47 -0
  43. package/dist/cli/quickstartCommands.js.map +1 -0
  44. package/dist/cli/shieldCommands.d.ts +30 -0
  45. package/dist/cli/shieldCommands.js +212 -0
  46. package/dist/cli/shieldCommands.js.map +1 -0
  47. package/dist/cli/tuiCommands.d.ts +7 -0
  48. package/dist/cli/tuiCommands.js +33 -0
  49. package/dist/cli/tuiCommands.js.map +1 -0
  50. package/dist/config/profiles.js +26 -0
  51. package/dist/config/profiles.js.map +1 -1
  52. package/dist/context/ContextBudget.js +2 -2
  53. package/dist/core/GbrainRuntime.d.ts +25 -0
  54. package/dist/core/GbrainRuntime.js +270 -0
  55. package/dist/core/GbrainRuntime.js.map +1 -0
  56. package/dist/cortex/GovernanceMetrics.d.ts +66 -0
  57. package/dist/cortex/GovernanceMetrics.js +230 -0
  58. package/dist/cortex/GovernanceMetrics.js.map +1 -0
  59. package/dist/cortex/InstinctExtractor.d.ts +61 -0
  60. package/dist/cortex/InstinctExtractor.js +184 -0
  61. package/dist/cortex/InstinctExtractor.js.map +1 -0
  62. package/dist/cortex/InstinctStore.d.ts +54 -0
  63. package/dist/cortex/InstinctStore.js +266 -0
  64. package/dist/cortex/InstinctStore.js.map +1 -0
  65. package/dist/cortex/ReflexionEngine.d.ts +34 -0
  66. package/dist/cortex/ReflexionEngine.js +157 -0
  67. package/dist/cortex/ReflexionEngine.js.map +1 -0
  68. package/dist/cortex/SessionInjector.d.ts +44 -0
  69. package/dist/cortex/SessionInjector.js +127 -0
  70. package/dist/cortex/SessionInjector.js.map +1 -0
  71. package/dist/cortex/adapters/ClaudeAdapter.d.ts +17 -0
  72. package/dist/cortex/adapters/ClaudeAdapter.js +61 -0
  73. package/dist/cortex/adapters/ClaudeAdapter.js.map +1 -0
  74. package/dist/cortex/adapters/CodexAdapter.d.ts +10 -0
  75. package/dist/cortex/adapters/CodexAdapter.js +52 -0
  76. package/dist/cortex/adapters/CodexAdapter.js.map +1 -0
  77. package/dist/cortex/adapters/CursorAdapter.d.ts +10 -0
  78. package/dist/cortex/adapters/CursorAdapter.js +46 -0
  79. package/dist/cortex/adapters/CursorAdapter.js.map +1 -0
  80. package/dist/cortex/adapters/GeminiAdapter.d.ts +11 -0
  81. package/dist/cortex/adapters/GeminiAdapter.js +48 -0
  82. package/dist/cortex/adapters/GeminiAdapter.js.map +1 -0
  83. package/dist/env/EnvironmentDoctor.js +221 -5
  84. package/dist/env/EnvironmentDoctor.js.map +1 -1
  85. package/dist/eval/BenchmarkPublisher.d.ts +25 -0
  86. package/dist/eval/BenchmarkPublisher.js +27 -0
  87. package/dist/eval/BenchmarkPublisher.js.map +1 -0
  88. package/dist/guardrails/DependencyAuditor.js +10 -1
  89. package/dist/guardrails/DependencyAuditor.js.map +1 -1
  90. package/dist/memory/MemoryProviders.js +38 -91
  91. package/dist/memory/MemoryProviders.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/runtime/ModelUsageLedger.d.ts +53 -2
  138. package/dist/runtime/ModelUsageLedger.js +243 -39
  139. package/dist/runtime/ModelUsageLedger.js.map +1 -1
  140. package/dist/setup/SetupVerification.d.ts +42 -0
  141. package/dist/setup/SetupVerification.js +180 -0
  142. package/dist/setup/SetupVerification.js.map +1 -0
  143. package/dist/shield/PolicyCompiler.d.ts +70 -0
  144. package/dist/shield/PolicyCompiler.js +540 -0
  145. package/dist/shield/PolicyCompiler.js.map +1 -0
  146. package/dist/shield/ProtectedPaths.d.ts +39 -0
  147. package/dist/shield/ProtectedPaths.js +179 -0
  148. package/dist/shield/ProtectedPaths.js.map +1 -0
  149. package/dist/shield/ShieldProtocol.d.ts +50 -0
  150. package/dist/shield/ShieldProtocol.js +103 -0
  151. package/dist/shield/ShieldProtocol.js.map +1 -0
  152. package/dist/skills/SkillMdStandard.d.ts +33 -0
  153. package/dist/skills/SkillMdStandard.js +88 -0
  154. package/dist/skills/SkillMdStandard.js.map +1 -0
  155. package/dist/skills/SkillRegistry.d.ts +9 -1
  156. package/dist/skills/SkillRegistry.js +20 -0
  157. package/dist/skills/SkillRegistry.js.map +1 -1
  158. package/dist/skills/interop/GStackInterop.d.ts +15 -0
  159. package/dist/skills/interop/GStackInterop.js +34 -0
  160. package/dist/skills/interop/GStackInterop.js.map +1 -0
  161. package/dist/skills/interop/OMCInterop.d.ts +15 -0
  162. package/dist/skills/interop/OMCInterop.js +34 -0
  163. package/dist/skills/interop/OMCInterop.js.map +1 -0
  164. package/dist/tools/ToolCapabilityRegistry.js +10 -0
  165. package/dist/tools/ToolCapabilityRegistry.js.map +1 -1
  166. package/dist/tui/TuiDashboard.d.ts +3 -0
  167. package/dist/tui/TuiDashboard.js +120 -0
  168. package/dist/tui/TuiDashboard.js.map +1 -0
  169. package/dist/workflow/GateCatalog.d.ts +2 -0
  170. package/dist/workflow/GateCatalog.js +59 -3
  171. package/dist/workflow/GateCatalog.js.map +1 -1
  172. package/dist/workflow/GovernanceTemplatePacks.d.ts +1 -1
  173. package/dist/workflow/GovernanceTemplatePacks.js +15 -0
  174. package/dist/workflow/GovernanceTemplatePacks.js.map +1 -1
  175. package/dist/workflow/TddLoop.d.ts +2 -0
  176. package/dist/workflow/TddLoop.js +2 -0
  177. package/dist/workflow/TddLoop.js.map +1 -1
  178. package/dist/workflow/UpgradeManager.d.ts +10 -1
  179. package/dist/workflow/UpgradeManager.js +55 -0
  180. package/dist/workflow/UpgradeManager.js.map +1 -1
  181. package/dist/workflow/VerificationProfile.d.ts +8 -0
  182. package/dist/workflow/VerificationProfile.js +62 -1
  183. package/dist/workflow/VerificationProfile.js.map +1 -1
  184. package/dist/workflow/VerificationSchema.d.ts +46 -0
  185. package/dist/workflow/VerificationSchema.js +97 -0
  186. package/dist/workflow/VerificationSchema.js.map +1 -0
  187. package/dist/workflow/autofix/AutoFixEngine.d.ts +37 -0
  188. package/dist/workflow/autofix/AutoFixEngine.js +169 -0
  189. package/dist/workflow/autofix/AutoFixEngine.js.map +1 -0
  190. package/dist/workflow/execution/RalphEngine.d.ts +18 -0
  191. package/dist/workflow/execution/RalphEngine.js +22 -0
  192. package/dist/workflow/execution/RalphEngine.js.map +1 -1
  193. package/dist/workflow/gates/EnhancedGates.d.ts +74 -0
  194. package/dist/workflow/gates/EnhancedGates.js +653 -0
  195. package/dist/workflow/gates/EnhancedGates.js.map +1 -0
  196. package/dist/workflow/gates/GateSystem.d.ts +3 -0
  197. package/dist/workflow/gates/GateSystem.js +94 -1
  198. package/dist/workflow/gates/GateSystem.js.map +1 -1
  199. package/dist/workflow/types.d.ts +1 -1
  200. package/docs/README.md +3 -0
  201. package/docs/guides/DEVELOPMENT_WORKFLOW.md +28 -9
  202. package/docs/guides/GETTING_STARTED.md +19 -0
  203. package/docs/guides/MIGRATION.md +119 -0
  204. package/docs/start/quickstart.md +1 -0
  205. package/docs/workflow/GATES_AND_SCORE.md +34 -1
  206. package/docs/workflow/README.md +58 -10
  207. package/package.json +7 -18
  208. package/scripts/workflow/lib/gbrain-runtime.mjs +185 -0
  209. package/scripts/workflow/lib/report-output.mjs +107 -0
  210. package/scripts/workflow/provider-rehearsal.mjs +129 -48
  211. package/scripts/workflow/setup-smoke.mjs +142 -8
  212. package/docs/ACTIVE_SECURITY_VISUAL_GATES.md +0 -87
  213. package/docs/AI_ENGINEERING_OS_POSITIONING.md +0 -607
  214. package/docs/BACKGROUND_HUNTER.md +0 -62
  215. package/docs/CODE_INTELLIGENCE.md +0 -180
  216. package/docs/CONTEXT_BUDGET.md +0 -155
  217. package/docs/DEPENDENCY_AUDIT.md +0 -118
  218. package/docs/EVOLUTION_SHADOW_MODE.md +0 -63
  219. package/docs/GITLAB_FLOW.md +0 -125
  220. package/docs/GOVERNANCE_DASHBOARD.md +0 -85
  221. package/docs/MEMORY_BRAIN.md +0 -104
  222. package/docs/MEMORY_FABRIC.md +0 -161
  223. package/docs/RESOURCE_GOVERNANCE.md +0 -92
  224. package/docs/RUNTIME_EVIDENCE.md +0 -101
  225. package/docs/WORKFLOW_EVAL.md +0 -151
  226. package/image/wechat-public.jpg +0 -0
  227. package/image/wxPay.jpg +0 -0
  228. package/image/zfb.jpg +0 -0
@@ -0,0 +1,147 @@
1
+ // SCALE Orchestrator — Issue Tracker Adapter
2
+ // 对齐 Symphony: tracker → polling → candidate selection
3
+ // Abstract adapter for GitHub Issues, Linear, Jira, etc.
4
+ // ---------------------------------------------------------------------------
5
+ // GitHub Issues adapter
6
+ // ---------------------------------------------------------------------------
7
+ export class GitHubTrackerAdapter {
8
+ constructor(config) {
9
+ const defaults = {
10
+ activeStates: ['open', 'in_progress'],
11
+ terminalStates: ['resolved', 'closed', 'cancelled'],
12
+ priorityLabels: { 'priority:critical': 0, 'priority:high': 1, 'priority:medium': 2, 'priority:low': 3 },
13
+ };
14
+ this.config = { ...defaults, ...config };
15
+ }
16
+ async fetchCandidates() {
17
+ const repo = `${this.config.owner}/${this.config.repo}`;
18
+ const issues = [];
19
+ try {
20
+ // Use gh CLI for GitHub Issues
21
+ const { execSync } = await import('node:child_process');
22
+ const args = ['issue', 'list', '--repo', repo, '--state', 'open', '--json', 'number,title,body,state,labels,assignees,createdAt,updatedAt', '--limit', '50'];
23
+ const stdout = execSync(`gh ${args.join(' ')}`, { encoding: 'utf-8', timeout: 10000 });
24
+ const raw = JSON.parse(stdout);
25
+ for (const item of raw) {
26
+ const labels = item.labels?.map(l => l.name) ?? [];
27
+ const priority = this.computePriority(labels);
28
+ issues.push({
29
+ id: String(item.number),
30
+ title: item.title,
31
+ description: item.body ?? '',
32
+ state: this.mapState(item.state),
33
+ labels,
34
+ assignee: item.assignees?.[0]?.login,
35
+ priority,
36
+ createdAt: item.createdAt,
37
+ updatedAt: item.updatedAt,
38
+ blockedBy: this.extractBlockedBy(item.body ?? ''),
39
+ });
40
+ }
41
+ }
42
+ catch {
43
+ // gh CLI not available — return empty
44
+ }
45
+ return issues;
46
+ }
47
+ async updateState(issueId, state, _metadata) {
48
+ const repo = `${this.config.owner}/${this.config.repo}`;
49
+ try {
50
+ const { execSync } = await import('node:child_process');
51
+ const stateMap = { open: 'open', in_progress: 'open', resolved: 'closed', closed: 'closed', cancelled: 'closed' };
52
+ execSync(`gh issue edit ${issueId} --repo ${repo} --state ${stateMap[state] ?? 'open'}`, { timeout: 5000 });
53
+ }
54
+ catch { /* ignore */ }
55
+ }
56
+ async addComment(issueId, body) {
57
+ const repo = `${this.config.owner}/${this.config.repo}`;
58
+ try {
59
+ const { execSync } = await import('node:child_process');
60
+ execSync(`gh issue comment ${issueId} --repo ${repo} --body "${body.replace(/"/g, '\\"')}"`, { timeout: 5000 });
61
+ }
62
+ catch { /* ignore */ }
63
+ }
64
+ async exists(issueId) {
65
+ const repo = `${this.config.owner}/${this.config.repo}`;
66
+ try {
67
+ const { execSync } = await import('node:child_process');
68
+ execSync(`gh issue view ${issueId} --repo ${repo} --json number`, { timeout: 5000 });
69
+ return true;
70
+ }
71
+ catch {
72
+ return false;
73
+ }
74
+ }
75
+ async getIssue(issueId) {
76
+ const repo = `${this.config.owner}/${this.config.repo}`;
77
+ try {
78
+ const { execSync } = await import('node:child_process');
79
+ const stdout = execSync(`gh issue view ${issueId} --repo ${repo} --json number,title,body,state,labels,assignees,createdAt,updatedAt`, { encoding: 'utf-8', timeout: 5000 });
80
+ const item = JSON.parse(stdout);
81
+ const labels = item.labels?.map((l) => l.name) ?? [];
82
+ return {
83
+ id: String(item.number),
84
+ title: item.title,
85
+ description: item.body ?? '',
86
+ state: this.mapState(item.state),
87
+ labels,
88
+ assignee: item.assignees?.[0]?.login,
89
+ priority: this.computePriority(labels),
90
+ createdAt: item.createdAt,
91
+ updatedAt: item.updatedAt,
92
+ blockedBy: this.extractBlockedBy(item.body ?? ''),
93
+ };
94
+ }
95
+ catch {
96
+ return null;
97
+ }
98
+ }
99
+ mapState(githubState) {
100
+ switch (githubState) {
101
+ case 'OPEN': return 'open';
102
+ case 'CLOSED': return 'closed';
103
+ default: return 'open';
104
+ }
105
+ }
106
+ computePriority(labels) {
107
+ for (const [label, prio] of Object.entries(this.config.priorityLabels)) {
108
+ if (labels.some(l => l.toLowerCase() === label.toLowerCase()))
109
+ return prio;
110
+ }
111
+ return 2; // default medium
112
+ }
113
+ extractBlockedBy(body) {
114
+ const match = body.match(/blocked[- ]by:\s*#?(\d+(?:,\s*#?\d+)*)/i);
115
+ if (match)
116
+ return match[1].split(',').map(s => s.replace('#', '').trim());
117
+ return [];
118
+ }
119
+ }
120
+ // ---------------------------------------------------------------------------
121
+ // Mock adapter for testing
122
+ // ---------------------------------------------------------------------------
123
+ export class MockTrackerAdapter {
124
+ constructor(issues = []) {
125
+ this.config = {
126
+ type: 'mock',
127
+ activeStates: ['open', 'in_progress'],
128
+ terminalStates: ['resolved', 'closed', 'cancelled'],
129
+ priorityLabels: {},
130
+ };
131
+ this.issues = new Map();
132
+ for (const issue of issues)
133
+ this.issues.set(issue.id, issue);
134
+ }
135
+ async fetchCandidates() {
136
+ return Array.from(this.issues.values()).filter(i => this.config.activeStates.includes(i.state));
137
+ }
138
+ async updateState(issueId, state) {
139
+ const issue = this.issues.get(issueId);
140
+ if (issue)
141
+ issue.state = state;
142
+ }
143
+ async addComment(_issueId, _body) { }
144
+ async exists(issueId) { return this.issues.has(issueId); }
145
+ async getIssue(issueId) { return this.issues.get(issueId) ?? null; }
146
+ }
147
+ //# sourceMappingURL=TrackerAdapter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TrackerAdapter.js","sourceRoot":"","sources":["../../src/orchestrator/TrackerAdapter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,66 @@
1
+ import type { OrchestratorPolicy } from './PolicyLoader.js';
2
+ export interface WorkspaceState {
3
+ id: string;
4
+ issueId: string;
5
+ branch: string;
6
+ path: string;
7
+ createdAt: string;
8
+ lastActivityAt: string;
9
+ agentPid?: number;
10
+ status: 'active' | 'idle' | 'terminal';
11
+ turnsCompleted: number;
12
+ }
13
+ export interface WorkspaceCreateResult {
14
+ success: boolean;
15
+ path: string;
16
+ branch: string;
17
+ error?: string;
18
+ }
19
+ export declare class WorkspaceManager {
20
+ private workspaces;
21
+ private policy;
22
+ constructor(policy: OrchestratorPolicy);
23
+ /**
24
+ * Create a git worktree for an issue.
25
+ */
26
+ create(issueId: string, baseBranch?: string): WorkspaceCreateResult;
27
+ /**
28
+ * Run before agent dispatch.
29
+ */
30
+ beforeRun(issueId: string): boolean;
31
+ /**
32
+ * Run after agent completes a turn.
33
+ */
34
+ afterRun(issueId: string): void;
35
+ /**
36
+ * Clean up and remove a worktree.
37
+ */
38
+ remove(issueId: string, force?: boolean): boolean;
39
+ /**
40
+ * Remove worktrees in terminal state.
41
+ */
42
+ cleanupTerminal(): number;
43
+ /**
44
+ * Verify safety invariants for a workspace.
45
+ */
46
+ verifySafety(workspacePath: string, agentCwd?: string): {
47
+ safe: boolean;
48
+ violations: string[];
49
+ };
50
+ /**
51
+ * Get all active workspaces.
52
+ */
53
+ listActive(): WorkspaceState[];
54
+ /**
55
+ * Get workspace count.
56
+ */
57
+ get activeCount(): number;
58
+ /**
59
+ * Check if we can create more workspaces.
60
+ */
61
+ get canCreate(): boolean;
62
+ /**
63
+ * Mark a workspace as idle or terminal.
64
+ */
65
+ updateStatus(issueId: string, status: WorkspaceState['status']): void;
66
+ }
@@ -0,0 +1,257 @@
1
+ // SCALE Orchestrator — Workspace Manager
2
+ // 对齐 Symphony: git worktree isolation + lifecycle hooks + safety invariants
3
+ // Path: <workspace.root>/<sanitized_issue_identifier>
4
+ // Safety: agent cwd ⊆ workspace, workspace ⊆ root, name sanitation
5
+ import { existsSync, mkdirSync, rmSync } from 'node:fs';
6
+ import { join, resolve, normalize, basename } from 'node:path';
7
+ import { execSync } from 'node:child_process';
8
+ import { logger } from '../core/logger.js';
9
+ // ---------------------------------------------------------------------------
10
+ // Safety invariants (aligned with Symphony)
11
+ // ---------------------------------------------------------------------------
12
+ const ALLOWED_CHARS_RE = /^[A-Za-z0-9._-]+$/;
13
+ function sanitizeName(name) {
14
+ // Replace any chars NOT in the allowlist with underscores
15
+ const sanitized = name.replace(/[^A-Za-z0-9._-]/g, '_');
16
+ if (!sanitized || sanitized.length > 64) {
17
+ return `ws-${Date.now().toString(36)}`;
18
+ }
19
+ return sanitized;
20
+ }
21
+ function assertSafety(workspacePath, root, expectedAgentCwd) {
22
+ const normalizedWs = normalize(resolve(workspacePath));
23
+ const normalizedRoot = normalize(resolve(root));
24
+ // Invariant 1: workspace path must be under configured root
25
+ if (!normalizedWs.startsWith(normalizedRoot)) {
26
+ throw new Error(`SAFETY: Workspace "${normalizedWs}" is outside root "${normalizedRoot}"`);
27
+ }
28
+ // Invariant 2: workspace name must pass sanitization
29
+ const name = basename(normalizedWs);
30
+ if (!ALLOWED_CHARS_RE.test(name)) {
31
+ throw new Error(`SAFETY: Workspace name "${name}" contains disallowed characters`);
32
+ }
33
+ // Invariant 3: if agent cwd is known, it must be inside workspace
34
+ if (expectedAgentCwd) {
35
+ const normalizedCwd = normalize(resolve(expectedAgentCwd));
36
+ if (!normalizedCwd.startsWith(normalizedWs)) {
37
+ throw new Error(`SAFETY: Agent cwd "${normalizedCwd}" is outside workspace "${normalizedWs}"`);
38
+ }
39
+ }
40
+ }
41
+ // ---------------------------------------------------------------------------
42
+ // WorkspaceManager
43
+ // ---------------------------------------------------------------------------
44
+ export class WorkspaceManager {
45
+ constructor(policy) {
46
+ this.workspaces = new Map();
47
+ this.policy = policy;
48
+ }
49
+ /**
50
+ * Create a git worktree for an issue.
51
+ */
52
+ create(issueId, baseBranch = 'master') {
53
+ const root = resolve(this.policy.workspace.root);
54
+ if (!existsSync(root))
55
+ mkdirSync(root, { recursive: true });
56
+ const safeId = sanitizeName(issueId);
57
+ const branch = `scale/${safeId}-${Date.now().toString(36)}`;
58
+ const wsPath = join(root, safeId);
59
+ // Safety check before creation
60
+ try {
61
+ assertSafety(wsPath, root);
62
+ }
63
+ catch (err) {
64
+ return { success: false, path: wsPath, branch, error: err.message };
65
+ }
66
+ // If worktree already exists, reuse it
67
+ if (existsSync(wsPath)) {
68
+ logger.info({ issueId, path: wsPath }, 'Worktree already exists, reusing');
69
+ const ws = this.workspaces.get(safeId);
70
+ if (ws)
71
+ return { success: true, path: wsPath, branch: ws.branch };
72
+ return { success: true, path: wsPath, branch };
73
+ }
74
+ try {
75
+ // Run lifecycle hook: after_create
76
+ if (this.policy.hooks.afterCreate) {
77
+ execSync(this.policy.hooks.afterCreate, {
78
+ env: { ...process.env, SCALE_ISSUE_ID: issueId, SCALE_WORKSPACE: wsPath, SCALE_BRANCH: branch },
79
+ timeout: 30000,
80
+ });
81
+ }
82
+ // Create git worktree
83
+ execSync(`git worktree add "${wsPath}" -b "${branch}"`, {
84
+ cwd: process.cwd(),
85
+ timeout: 30000,
86
+ });
87
+ const ws = {
88
+ id: safeId,
89
+ issueId,
90
+ branch,
91
+ path: wsPath,
92
+ createdAt: new Date().toISOString(),
93
+ lastActivityAt: new Date().toISOString(),
94
+ status: 'active',
95
+ turnsCompleted: 0,
96
+ };
97
+ this.workspaces.set(safeId, ws);
98
+ logger.info({ issueId, path: wsPath, branch }, 'Worktree created');
99
+ return { success: true, path: wsPath, branch };
100
+ }
101
+ catch (err) {
102
+ logger.error({ err, issueId, path: wsPath }, 'Failed to create worktree');
103
+ return { success: false, path: wsPath, branch, error: String(err) };
104
+ }
105
+ }
106
+ /**
107
+ * Run before agent dispatch.
108
+ */
109
+ beforeRun(issueId) {
110
+ const safeId = sanitizeName(issueId);
111
+ const ws = this.workspaces.get(safeId);
112
+ if (!ws)
113
+ return false;
114
+ try {
115
+ if (this.policy.hooks.beforeRun) {
116
+ execSync(this.policy.hooks.beforeRun, {
117
+ env: { ...process.env, SCALE_ISSUE_ID: issueId, SCALE_WORKSPACE: ws.path },
118
+ timeout: 10000,
119
+ });
120
+ }
121
+ ws.lastActivityAt = new Date().toISOString();
122
+ this.workspaces.set(safeId, ws);
123
+ return true;
124
+ }
125
+ catch (err) {
126
+ logger.warn({ err, issueId }, 'beforeRun hook failed');
127
+ return false;
128
+ }
129
+ }
130
+ /**
131
+ * Run after agent completes a turn.
132
+ */
133
+ afterRun(issueId) {
134
+ const safeId = sanitizeName(issueId);
135
+ const ws = this.workspaces.get(safeId);
136
+ if (!ws)
137
+ return;
138
+ ws.turnsCompleted++;
139
+ ws.lastActivityAt = new Date().toISOString();
140
+ if (this.policy.hooks.afterRun) {
141
+ try {
142
+ execSync(this.policy.hooks.afterRun, {
143
+ env: { ...process.env, SCALE_ISSUE_ID: issueId, SCALE_WORKSPACE: ws.path, SCALE_TURN: String(ws.turnsCompleted) },
144
+ timeout: 10000,
145
+ });
146
+ }
147
+ catch { /* best-effort */ }
148
+ }
149
+ this.workspaces.set(safeId, ws);
150
+ }
151
+ /**
152
+ * Clean up and remove a worktree.
153
+ */
154
+ remove(issueId, force = false) {
155
+ const safeId = sanitizeName(issueId);
156
+ const ws = this.workspaces.get(safeId);
157
+ if (!ws)
158
+ return false;
159
+ try {
160
+ if (this.policy.hooks.beforeRemove) {
161
+ execSync(this.policy.hooks.beforeRemove, {
162
+ env: { ...process.env, SCALE_ISSUE_ID: issueId, SCALE_WORKSPACE: ws.path, SCALE_BRANCH: ws.branch },
163
+ timeout: 10000,
164
+ });
165
+ }
166
+ }
167
+ catch { /* best effort */ }
168
+ try {
169
+ execSync(`git worktree remove "${ws.path}" ${force ? '--force' : ''}`, {
170
+ cwd: process.cwd(),
171
+ timeout: 15000,
172
+ });
173
+ }
174
+ catch {
175
+ // If git worktree remove fails, try manual cleanup
176
+ if (force) {
177
+ try {
178
+ rmSync(ws.path, { recursive: true, force: true });
179
+ }
180
+ catch { /* ignore */ }
181
+ }
182
+ else {
183
+ return false;
184
+ }
185
+ }
186
+ this.workspaces.delete(safeId);
187
+ logger.info({ issueId, path: ws.path }, 'Worktree removed');
188
+ return true;
189
+ }
190
+ /**
191
+ * Remove worktrees in terminal state.
192
+ */
193
+ cleanupTerminal() {
194
+ let cleaned = 0;
195
+ for (const [id, ws] of this.workspaces) {
196
+ if (ws.status === 'terminal') {
197
+ if (this.remove(ws.issueId, true))
198
+ cleaned++;
199
+ }
200
+ }
201
+ // Also clean old workspaces by age
202
+ const maxAgeMs = this.policy.workspace.maxWorkspaceAgeHours * 60 * 60 * 1000;
203
+ const now = Date.now();
204
+ for (const [id, ws] of this.workspaces) {
205
+ const age = now - new Date(ws.lastActivityAt).getTime();
206
+ if (age > maxAgeMs && ws.status === 'idle') {
207
+ ws.status = 'terminal';
208
+ if (this.remove(ws.issueId, true))
209
+ cleaned++;
210
+ }
211
+ }
212
+ return cleaned;
213
+ }
214
+ /**
215
+ * Verify safety invariants for a workspace.
216
+ */
217
+ verifySafety(workspacePath, agentCwd) {
218
+ try {
219
+ assertSafety(workspacePath, this.policy.workspace.root, agentCwd);
220
+ return { safe: true, violations: [] };
221
+ }
222
+ catch (err) {
223
+ return { safe: false, violations: [err.message] };
224
+ }
225
+ }
226
+ /**
227
+ * Get all active workspaces.
228
+ */
229
+ listActive() {
230
+ return Array.from(this.workspaces.values()).filter(w => w.status !== 'terminal');
231
+ }
232
+ /**
233
+ * Get workspace count.
234
+ */
235
+ get activeCount() {
236
+ return this.listActive().length;
237
+ }
238
+ /**
239
+ * Check if we can create more workspaces.
240
+ */
241
+ get canCreate() {
242
+ return this.activeCount < this.policy.polling.maxParallelWorkspaces;
243
+ }
244
+ /**
245
+ * Mark a workspace as idle or terminal.
246
+ */
247
+ updateStatus(issueId, status) {
248
+ const safeId = sanitizeName(issueId);
249
+ const ws = this.workspaces.get(safeId);
250
+ if (ws) {
251
+ ws.status = status;
252
+ ws.lastActivityAt = new Date().toISOString();
253
+ this.workspaces.set(safeId, ws);
254
+ }
255
+ }
256
+ }
257
+ //# sourceMappingURL=WorkspaceManager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"WorkspaceManager.js","sourceRoot":"","sources":["../../src/orchestrator/WorkspaceManager.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,23 @@
1
+ export interface DaemonStatus {
2
+ running: boolean;
3
+ pid?: number;
4
+ startedAt?: string;
5
+ requestsProcessed: number;
6
+ idleSeconds: number;
7
+ }
8
+ export declare class BrowserDaemon {
9
+ private browser;
10
+ private requestsProcessed;
11
+ private lastActivity;
12
+ private idleTimer;
13
+ start(): Promise<{
14
+ success: boolean;
15
+ pid: number;
16
+ }>;
17
+ stop(): Promise<{
18
+ success: boolean;
19
+ }>;
20
+ status(): DaemonStatus;
21
+ getBrowser(): unknown;
22
+ private resetIdleTimer;
23
+ }
@@ -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"}