@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.
- package/README.md +30 -2
- package/dist/api/cli.js +286 -7
- package/dist/api/cli.js.map +1 -1
- package/dist/api/doctor.js +1 -1
- package/dist/api/doctor.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/bootstrap/DependencyBootstrap.d.ts +1 -0
- package/dist/bootstrap/DependencyBootstrap.js +137 -25
- package/dist/bootstrap/DependencyBootstrap.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/capabilities/InstalledSkillsIntegration.js +29 -9
- package/dist/capabilities/InstalledSkillsIntegration.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/config/profiles.js +26 -0
- package/dist/config/profiles.js.map +1 -1
- package/dist/context/ContextBudget.js +2 -2
- package/dist/core/GbrainRuntime.d.ts +25 -0
- package/dist/core/GbrainRuntime.js +270 -0
- package/dist/core/GbrainRuntime.js.map +1 -0
- 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/env/EnvironmentDoctor.js +221 -5
- package/dist/env/EnvironmentDoctor.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/memory/MemoryProviders.js +38 -91
- package/dist/memory/MemoryProviders.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/runtime/ModelUsageLedger.d.ts +53 -2
- package/dist/runtime/ModelUsageLedger.js +243 -39
- package/dist/runtime/ModelUsageLedger.js.map +1 -1
- package/dist/setup/SetupVerification.d.ts +42 -0
- package/dist/setup/SetupVerification.js +180 -0
- package/dist/setup/SetupVerification.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/tools/ToolCapabilityRegistry.js +10 -0
- package/dist/tools/ToolCapabilityRegistry.js.map +1 -1
- 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 +62 -1
- 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/start/quickstart.md +1 -0
- package/docs/workflow/GATES_AND_SCORE.md +34 -1
- package/docs/workflow/README.md +58 -10
- package/package.json +7 -18
- package/scripts/workflow/lib/gbrain-runtime.mjs +185 -0
- package/scripts/workflow/lib/report-output.mjs +107 -0
- package/scripts/workflow/provider-rehearsal.mjs +129 -48
- package/scripts/workflow/setup-smoke.mjs +142 -8
- 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 -155
- 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 -85
- 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,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":"AAAA,6CAA6C;AAC7C,uDAAuD;AACvD,yDAAyD;AAiDzD,8EAA8E;AAC9E,wBAAwB;AACxB,8EAA8E;AAE9E,MAAM,OAAO,oBAAoB;IAG/B,YAAY,MAAqB;QAC/B,MAAM,QAAQ,GAA2B;YACvC,YAAY,EAAE,CAAC,MAAM,EAAE,aAAa,CAAC;YACrC,cAAc,EAAE,CAAC,UAAU,EAAE,QAAQ,EAAE,WAAW,CAAC;YACnD,cAAc,EAAE,EAAE,mBAAmB,EAAE,CAAC,EAAE,eAAe,EAAE,CAAC,EAAE,iBAAiB,EAAE,CAAC,EAAE,cAAc,EAAE,CAAC,EAAE;SACxG,CAAA;QACD,IAAI,CAAC,MAAM,GAAG,EAAE,GAAG,QAAQ,EAAE,GAAG,MAAM,EAAmB,CAAA;IAC3D,CAAC;IAED,KAAK,CAAC,eAAe;QACnB,MAAM,IAAI,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAA;QACvD,MAAM,MAAM,GAAmB,EAAE,CAAA;QAEjC,IAAI,CAAC;YACH,+BAA+B;YAC/B,MAAM,EAAE,QAAQ,EAAE,GAAG,MAAM,MAAM,CAAC,oBAAoB,CAAC,CAAA;YACvD,MAAM,IAAI,GAAG,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,8DAA8D,EAAE,SAAS,EAAE,IAAI,CAAC,CAAA;YAC5J,MAAM,MAAM,GAAG,QAAQ,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAA;YACtF,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAI3B,CAAA;YAEF,KAAK,MAAM,IAAI,IAAI,GAAG,EAAE,CAAC;gBACvB,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,CAAA;gBAClD,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,CAAA;gBAE7C,MAAM,CAAC,IAAI,CAAC;oBACV,EAAE,EAAE,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC;oBACvB,KAAK,EAAE,IAAI,CAAC,KAAK;oBACjB,WAAW,EAAE,IAAI,CAAC,IAAI,IAAI,EAAE;oBAC5B,KAAK,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC;oBAChC,MAAM;oBACN,QAAQ,EAAE,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK;oBACpC,QAAQ;oBACR,SAAS,EAAE,IAAI,CAAC,SAAS;oBACzB,SAAS,EAAE,IAAI,CAAC,SAAS;oBACzB,SAAS,EAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,IAAI,EAAE,CAAC;iBAClD,CAAC,CAAA;YACJ,CAAC;QACH,CAAC;QAAC,MAAM,CAAC;YACP,sCAAsC;QACxC,CAAC;QAED,OAAO,MAAM,CAAA;IACf,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,OAAe,EAAE,KAAiB,EAAE,SAAmC;QACvF,MAAM,IAAI,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAA;QACvD,IAAI,CAAC;YACH,MAAM,EAAE,QAAQ,EAAE,GAAG,MAAM,MAAM,CAAC,oBAAoB,CAAC,CAAA;YACvD,MAAM,QAAQ,GAA2B,EAAE,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,SAAS,EAAE,QAAQ,EAAE,CAAA;YACzI,QAAQ,CAAC,iBAAiB,OAAO,WAAW,IAAI,YAAY,QAAQ,CAAC,KAAK,CAAC,IAAI,MAAM,EAAE,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAA;QAC7G,CAAC;QAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,UAAU,CAAC,OAAe,EAAE,IAAY;QAC5C,MAAM,IAAI,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAA;QACvD,IAAI,CAAC;YACH,MAAM,EAAE,QAAQ,EAAE,GAAG,MAAM,MAAM,CAAC,oBAAoB,CAAC,CAAA;YACvD,QAAQ,CAAC,oBAAoB,OAAO,WAAW,IAAI,YAAY,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,KAAK,CAAC,GAAG,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAA;QACjH,CAAC;QAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,MAAM,CAAC,OAAe;QAC1B,MAAM,IAAI,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAA;QACvD,IAAI,CAAC;YACH,MAAM,EAAE,QAAQ,EAAE,GAAG,MAAM,MAAM,CAAC,oBAAoB,CAAC,CAAA;YACvD,QAAQ,CAAC,iBAAiB,OAAO,WAAW,IAAI,gBAAgB,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAA;YACpF,OAAO,IAAI,CAAA;QACb,CAAC;QAAC,MAAM,CAAC;YAAC,OAAO,KAAK,CAAA;QAAC,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,QAAQ,CAAC,OAAe;QAC5B,MAAM,IAAI,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAA;QACvD,IAAI,CAAC;YACH,MAAM,EAAE,QAAQ,EAAE,GAAG,MAAM,MAAM,CAAC,oBAAoB,CAAC,CAAA;YACvD,MAAM,MAAM,GAAG,QAAQ,CAAC,iBAAiB,OAAO,WAAW,IAAI,sEAAsE,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAA;YAC5K,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAA;YAC/B,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAmB,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,CAAA;YACtE,OAAO;gBACL,EAAE,EAAE,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC;gBACvB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,WAAW,EAAE,IAAI,CAAC,IAAI,IAAI,EAAE;gBAC5B,KAAK,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC;gBAChC,MAAM;gBACN,QAAQ,EAAE,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK;gBACpC,QAAQ,EAAE,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC;gBACtC,SAAS,EAAE,IAAI,CAAC,SAAS;gBACzB,SAAS,EAAE,IAAI,CAAC,SAAS;gBACzB,SAAS,EAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,IAAI,EAAE,CAAC;aAClD,CAAA;QACH,CAAC;QAAC,MAAM,CAAC;YAAC,OAAO,IAAI,CAAA;QAAC,CAAC;IACzB,CAAC;IAEO,QAAQ,CAAC,WAAmB;QAClC,QAAQ,WAAW,EAAE,CAAC;YACpB,KAAK,MAAM,CAAC,CAAC,OAAO,MAAM,CAAA;YAC1B,KAAK,QAAQ,CAAC,CAAC,OAAO,QAAQ,CAAA;YAC9B,OAAO,CAAC,CAAC,OAAO,MAAM,CAAA;QACxB,CAAC;IACH,CAAC;IAEO,eAAe,CAAC,MAAgB;QACtC,KAAK,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,EAAE,CAAC;YACvE,IAAI,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,KAAK,KAAK,CAAC,WAAW,EAAE,CAAC;gBAAE,OAAO,IAAI,CAAA;QAC5E,CAAC;QACD,OAAO,CAAC,CAAA,CAAC,iBAAiB;IAC5B,CAAC;IAEO,gBAAgB,CAAC,IAAY;QACnC,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,yCAAyC,CAAC,CAAA;QACnE,IAAI,KAAK;YAAE,OAAO,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC,CAAA;QACzE,OAAO,EAAE,CAAA;IACX,CAAC;CACF;AAED,8EAA8E;AAC9E,2BAA2B;AAC3B,8EAA8E;AAE9E,MAAM,OAAO,kBAAkB;IAS7B,YAAY,SAAyB,EAAE;QAR9B,WAAM,GAAkB;YAC/B,IAAI,EAAE,MAAM;YACZ,YAAY,EAAE,CAAC,MAAM,EAAE,aAAa,CAAC;YACrC,cAAc,EAAE,CAAC,UAAU,EAAE,QAAQ,EAAE,WAAW,CAAC;YACnD,cAAc,EAAE,EAAE;SACnB,CAAA;QACO,WAAM,GAA8B,IAAI,GAAG,EAAE,CAAA;QAGnD,KAAK,MAAM,KAAK,IAAI,MAAM;YAAE,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,EAAE,KAAK,CAAC,CAAA;IAC9D,CAAC;IAED,KAAK,CAAC,eAAe;QACnB,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CACjD,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAA;IAC/C,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,OAAe,EAAE,KAAiB;QAClD,MAAM,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,CAAA;QACtC,IAAI,KAAK;YAAE,KAAK,CAAC,KAAK,GAAG,KAAK,CAAA;IAChC,CAAC;IAED,KAAK,CAAC,UAAU,CAAC,QAAgB,EAAE,KAAa,IAA8B,CAAC;IAC/E,KAAK,CAAC,MAAM,CAAC,OAAe,IAAsB,OAAO,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,CAAA,CAAC,CAAC;IACnF,KAAK,CAAC,QAAQ,CAAC,OAAe,IAAkC,OAAO,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,IAAI,CAAA,CAAC,CAAC;CAC1G"}
|
|
@@ -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"}
|