sdd-full 4.2.0 → 4.3.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/bin.js +31 -63
- package/package.json +1 -1
- package/skills/README.md +97 -0
- package/skills/call-adaptation/SKILL.md +23 -0
- package/skills/call-adaptation/call-adaptation-guide.md +136 -0
- package/skills/call-adaptation/claude-code-call-spec.md +50 -0
- package/skills/call-adaptation/trae-call-spec.md +56 -0
- package/skills/checklist.md +154 -0
- package/skills/design-planning/ai-coding-rules/SKILL.md +52 -0
- package/skills/design-planning/design-to-code/SKILL.md +53 -0
- package/skills/design-planning/enterprise-spec/SKILL.md +52 -3
- package/skills/design-planning/flutter-av/SKILL.md +44 -34
- package/skills/design-planning/flutter-map/SKILL.md +41 -31
- package/skills/design-planning/function-sdd/SKILL.md +54 -0
- package/skills/design-planning/sdd-code/SKILL.md +347 -0
- package/skills/design-planning/sdd-deploy/SKILL.md +501 -0
- package/skills/design-planning/sdd-ops/SKILL.md +306 -0
- package/skills/design-planning/sdd-test/SKILL.md +383 -0
- package/skills/design-planning/ui-sdd/SKILL.md +291 -0
- package/skills/design-planning/ui-sdd-specialized/SKILL.md +46 -40
- package/skills/design-planning/writing-plans/SKILL.md +144 -0
- package/skills/development-execution/flutter-errors/SKILL.md +44 -34
- package/skills/development-execution/sdd-add/SKILL.md +540 -0
- package/skills/development-execution/systematic-debugging/SKILL.md +298 -0
- package/skills/development-execution/test-driven-development/SKILL.md +373 -0
- package/skills/development-execution/verification-before-completion/SKILL.md +141 -0
- package/skills/knowledge-precipitation/claudeception/SKILL.md +96 -0
- package/skills/knowledge-precipitation/mempalace-auto-saver/SKILL.md +302 -0
- package/skills/quality-assurance/bdd-acceptance/SKILL.md +44 -37
- package/skills/quality-assurance/flutter-test/SKILL.md +56 -0
- package/skills/quality-assurance/quality-gate/SKILL.md +350 -0
- package/skills/quality-assurance/security-audit/SKILL.md +386 -0
- package/skills/release-ops/finishing-a-development-branch/SKILL.md +202 -0
- package/skills/release-ops/release-flow/SKILL.md +404 -0
- package/skills/requirement-analysis/brainstorming/SKILL.md +166 -0
- package/skills/requirement-analysis/competitive-brief/SKILL.md +121 -0
- package/skills/requirement-analysis/market-research/SKILL.md +143 -0
- package/skills/requirement-analysis/prd-write/SKILL.md +111 -0
- package/skills/requirement-analysis/requirement-completion-officer/SKILL.md +124 -0
- package/skills/requirement-analysis/sdd/SKILL.md +1044 -0
- package/skills/requirement-analysis/sdd-full/SKILL.md +717 -36
- package/skills/requirement-analysis/unified-flow/SKILL.md +128 -26
- package/skills/rules/project_rules.md +167 -0
- package/skills/rules/user_rules.md +254 -69
- package/skills/special-tools/env-check/SKILL.md +40 -34
- package/skills/special-tools/receiving-code-review/SKILL.md +215 -0
- package/skills/special-tools/requesting-code-review/SKILL.md +107 -0
- package/skills/special-tools/using-superpowers/SKILL.md +117 -0
- package/skills/templates/API-SDD.md +31 -0
- package/skills/templates/Andrej Karpathy AI/347/274/226/347/240/201/350/247/204/345/210/231/350/220/275/345/234/260SDD.md" +117 -0
- package/skills/templates/BDD/351/243/216/346/240/274/351/252/214/346/224/266/346/240/207/345/207/206SDD.md +147 -0
- package/skills/templates/Base-SDD.md +38 -0
- package/skills/templates/Brain-SDD.md +36 -0
- package/skills/templates/Code-SDD.md +41 -0
- package/skills/templates/Competitor-SDD.md +34 -0
- package/skills/templates/Env-SDD.md +37 -0
- package/skills/templates/Flutter/345/205/250/347/261/273/345/236/213/346/265/213/350/257/225/347/255/226/347/225/245SDD.md +162 -0
- package/skills/templates/Flutter/345/234/260/345/233/276/345/257/274/350/210/252/344/270/232/345/212/241SDD.md +136 -0
- package/skills/templates/Flutter/345/270/270/350/247/201/345/274/202/345/270/270/344/270/223/351/241/271SDD.md +159 -0
- package/skills/templates/Flutter/351/237/263/350/247/206/351/242/221/345/205/250/346/240/210SDD.md +121 -0
- package/skills/templates/PRD-SDD.md +45 -0
- package/skills/templates/SKILL.md +91 -0
- package/skills/templates/Test-SDD.md +34 -0
- package/skills/templates/UI-SDD.md +38 -0
- package/skills/templates/UI-SDD/344/270/223/347/224/250/346/250/241/346/235/277.md +141 -0
- package/skills/templates/UI/350/265/204/346/272/220/346/217/220/347/244/272/350/257/215/347/224/237/346/210/220SDD.md +67 -0
- package/skills/templates//344/274/201/344/270/232/347/272/247/345/205/250/346/240/210/345/267/245/347/250/213/350/247/204/350/214/203SDD.md +152 -0
- package/skills/templates//345/205/250/346/265/201/347/250/213SDD/350/236/215/345/220/210/344/275/223/347/263/273.md +198 -0
- package/skills/templates//345/212/237/350/203/275SDD/344/270/223/347/224/250/346/250/241/346/235/277.md +132 -0
- package/skills/templates//347/216/257/345/242/203/351/242/204/346/243/200/346/240/207/345/207/206/345/214/226SDD.md +153 -0
- package/skills/templates//351/253/230/344/277/235/347/234/237/350/256/276/350/256/241/350/275/254/344/273/243/347/240/201SDD.md +119 -0
- package/skills//345/256/214/346/225/264/345/274/200/345/217/221/346/265/201/347/250/213/346/211/213/345/206/214.md +408 -0
- package/skills//346/212/200/350/203/275/344/275/223/347/263/273/345/256/214/345/226/204/345/273/272/350/256/256.md +305 -0
- package/skills//346/212/200/350/203/275/344/275/277/347/224/250/346/214/207/345/215/227.md +285 -0
- package/skills//346/212/200/350/203/275/345/206/263/347/255/226/346/240/221.md +320 -0
- package/skills/brainstorming/SKILL.md +0 -164
- package/skills/brainstorming/scripts/frame-template.html +0 -214
- package/skills/brainstorming/scripts/helper.js +0 -88
- package/skills/brainstorming/scripts/server.cjs +0 -338
- package/skills/brainstorming/scripts/start-server.sh +0 -153
- package/skills/brainstorming/scripts/stop-server.sh +0 -55
- package/skills/brainstorming/spec-document-reviewer-prompt.md +0 -48
- package/skills/brainstorming/visual-companion.md +0 -286
- package/skills/chinese-code-review/SKILL.md +0 -277
- package/skills/chinese-commit-conventions/SKILL.md +0 -364
- package/skills/chinese-documentation/SKILL.md +0 -448
- package/skills/chinese-git-workflow/SKILL.md +0 -510
- package/skills/dispatching-parallel-agents/SKILL.md +0 -182
- package/skills/executing-plans/SKILL.md +0 -175
- package/skills/finishing-a-development-branch/SKILL.md +0 -200
- package/skills/mcp-builder/SKILL.md +0 -255
- package/skills/receiving-code-review/SKILL.md +0 -213
- package/skills/requesting-code-review/SKILL.md +0 -105
- package/skills/requesting-code-review/code-reviewer.md +0 -146
- package/skills/rules/skill-map.md +0 -97
- package/skills/subagent-driven-development/SKILL.md +0 -277
- package/skills/subagent-driven-development/code-quality-reviewer-prompt.md +0 -26
- package/skills/subagent-driven-development/implementer-prompt.md +0 -113
- package/skills/subagent-driven-development/spec-reviewer-prompt.md +0 -61
- package/skills/systematic-debugging/CREATION-LOG.md +0 -119
- package/skills/systematic-debugging/SKILL.md +0 -296
- package/skills/systematic-debugging/condition-based-waiting-example.ts +0 -158
- package/skills/systematic-debugging/condition-based-waiting.md +0 -115
- package/skills/systematic-debugging/defense-in-depth.md +0 -122
- package/skills/systematic-debugging/find-polluter.sh +0 -63
- package/skills/systematic-debugging/root-cause-tracing.md +0 -169
- package/skills/systematic-debugging/test-academic.md +0 -14
- package/skills/systematic-debugging/test-pressure-1.md +0 -58
- package/skills/systematic-debugging/test-pressure-2.md +0 -68
- package/skills/systematic-debugging/test-pressure-3.md +0 -69
- package/skills/test-driven-development/SKILL.md +0 -371
- package/skills/test-driven-development/testing-anti-patterns.md +0 -299
- package/skills/using-git-worktrees/SKILL.md +0 -218
- package/skills/using-superpowers/SKILL.md +0 -134
- package/skills/using-superpowers/references/codex-tools.md +0 -25
- package/skills/using-superpowers/references/gemini-tools.md +0 -33
- package/skills/verification-before-completion/SKILL.md +0 -139
- package/skills/workflow-runner/SKILL.md +0 -172
- package/skills/writing-plans/SKILL.md +0 -152
- package/skills/writing-plans/plan-document-reviewer-prompt.md +0 -49
- package/skills/writing-skills/SKILL.md +0 -654
- package/skills/writing-skills/anthropic-best-practices.md +0 -1149
- package/skills/writing-skills/examples/CLAUDE_MD_TESTING.md +0 -189
- package/skills/writing-skills/graphviz-conventions.dot +0 -172
- package/skills/writing-skills/persuasion-principles.md +0 -187
- package/skills/writing-skills/render-graphs.js +0 -168
- package/skills/writing-skills/testing-skills-with-subagents.md +0 -384
|
@@ -1,88 +0,0 @@
|
|
|
1
|
-
(function() {
|
|
2
|
-
const WS_URL = 'ws://' + window.location.host;
|
|
3
|
-
let ws = null;
|
|
4
|
-
let eventQueue = [];
|
|
5
|
-
|
|
6
|
-
function connect() {
|
|
7
|
-
ws = new WebSocket(WS_URL);
|
|
8
|
-
|
|
9
|
-
ws.onopen = () => {
|
|
10
|
-
eventQueue.forEach(e => ws.send(JSON.stringify(e)));
|
|
11
|
-
eventQueue = [];
|
|
12
|
-
};
|
|
13
|
-
|
|
14
|
-
ws.onmessage = (msg) => {
|
|
15
|
-
const data = JSON.parse(msg.data);
|
|
16
|
-
if (data.type === 'reload') {
|
|
17
|
-
window.location.reload();
|
|
18
|
-
}
|
|
19
|
-
};
|
|
20
|
-
|
|
21
|
-
ws.onclose = () => {
|
|
22
|
-
setTimeout(connect, 1000);
|
|
23
|
-
};
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
function sendEvent(event) {
|
|
27
|
-
event.timestamp = Date.now();
|
|
28
|
-
if (ws && ws.readyState === WebSocket.OPEN) {
|
|
29
|
-
ws.send(JSON.stringify(event));
|
|
30
|
-
} else {
|
|
31
|
-
eventQueue.push(event);
|
|
32
|
-
}
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
// Capture clicks on choice elements
|
|
36
|
-
document.addEventListener('click', (e) => {
|
|
37
|
-
const target = e.target.closest('[data-choice]');
|
|
38
|
-
if (!target) return;
|
|
39
|
-
|
|
40
|
-
sendEvent({
|
|
41
|
-
type: 'click',
|
|
42
|
-
text: target.textContent.trim(),
|
|
43
|
-
choice: target.dataset.choice,
|
|
44
|
-
id: target.id || null
|
|
45
|
-
});
|
|
46
|
-
|
|
47
|
-
// Update indicator bar (defer so toggleSelect runs first)
|
|
48
|
-
setTimeout(() => {
|
|
49
|
-
const indicator = document.getElementById('indicator-text');
|
|
50
|
-
if (!indicator) return;
|
|
51
|
-
const container = target.closest('.options') || target.closest('.cards');
|
|
52
|
-
const selected = container ? container.querySelectorAll('.selected') : [];
|
|
53
|
-
if (selected.length === 0) {
|
|
54
|
-
indicator.textContent = 'Click an option above, then return to the terminal';
|
|
55
|
-
} else if (selected.length === 1) {
|
|
56
|
-
const label = selected[0].querySelector('h3, .content h3, .card-body h3')?.textContent?.trim() || selected[0].dataset.choice;
|
|
57
|
-
indicator.innerHTML = '<span class="selected-text">' + label + ' selected</span> — return to terminal to continue';
|
|
58
|
-
} else {
|
|
59
|
-
indicator.innerHTML = '<span class="selected-text">' + selected.length + ' selected</span> — return to terminal to continue';
|
|
60
|
-
}
|
|
61
|
-
}, 0);
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
// Frame UI: selection tracking
|
|
65
|
-
window.selectedChoice = null;
|
|
66
|
-
|
|
67
|
-
window.toggleSelect = function(el) {
|
|
68
|
-
const container = el.closest('.options') || el.closest('.cards');
|
|
69
|
-
const multi = container && container.dataset.multiselect !== undefined;
|
|
70
|
-
if (container && !multi) {
|
|
71
|
-
container.querySelectorAll('.option, .card').forEach(o => o.classList.remove('selected'));
|
|
72
|
-
}
|
|
73
|
-
if (multi) {
|
|
74
|
-
el.classList.toggle('selected');
|
|
75
|
-
} else {
|
|
76
|
-
el.classList.add('selected');
|
|
77
|
-
}
|
|
78
|
-
window.selectedChoice = el.dataset.choice;
|
|
79
|
-
};
|
|
80
|
-
|
|
81
|
-
// Expose API for explicit use
|
|
82
|
-
window.brainstorm = {
|
|
83
|
-
send: sendEvent,
|
|
84
|
-
choice: (value, metadata = {}) => sendEvent({ type: 'choice', value, ...metadata })
|
|
85
|
-
};
|
|
86
|
-
|
|
87
|
-
connect();
|
|
88
|
-
})();
|
|
@@ -1,338 +0,0 @@
|
|
|
1
|
-
const crypto = require('crypto');
|
|
2
|
-
const http = require('http');
|
|
3
|
-
const fs = require('fs');
|
|
4
|
-
const path = require('path');
|
|
5
|
-
|
|
6
|
-
// ========== WebSocket Protocol (RFC 6455) ==========
|
|
7
|
-
|
|
8
|
-
const OPCODES = { TEXT: 0x01, CLOSE: 0x08, PING: 0x09, PONG: 0x0A };
|
|
9
|
-
const WS_MAGIC = '258EAFA5-E914-47DA-95CA-C5AB0DC85B11';
|
|
10
|
-
|
|
11
|
-
function computeAcceptKey(clientKey) {
|
|
12
|
-
return crypto.createHash('sha1').update(clientKey + WS_MAGIC).digest('base64');
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
function encodeFrame(opcode, payload) {
|
|
16
|
-
const fin = 0x80;
|
|
17
|
-
const len = payload.length;
|
|
18
|
-
let header;
|
|
19
|
-
|
|
20
|
-
if (len < 126) {
|
|
21
|
-
header = Buffer.alloc(2);
|
|
22
|
-
header[0] = fin | opcode;
|
|
23
|
-
header[1] = len;
|
|
24
|
-
} else if (len < 65536) {
|
|
25
|
-
header = Buffer.alloc(4);
|
|
26
|
-
header[0] = fin | opcode;
|
|
27
|
-
header[1] = 126;
|
|
28
|
-
header.writeUInt16BE(len, 2);
|
|
29
|
-
} else {
|
|
30
|
-
header = Buffer.alloc(10);
|
|
31
|
-
header[0] = fin | opcode;
|
|
32
|
-
header[1] = 127;
|
|
33
|
-
header.writeBigUInt64BE(BigInt(len), 2);
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
return Buffer.concat([header, payload]);
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
function decodeFrame(buffer) {
|
|
40
|
-
if (buffer.length < 2) return null;
|
|
41
|
-
|
|
42
|
-
const secondByte = buffer[1];
|
|
43
|
-
const opcode = buffer[0] & 0x0F;
|
|
44
|
-
const masked = (secondByte & 0x80) !== 0;
|
|
45
|
-
let payloadLen = secondByte & 0x7F;
|
|
46
|
-
let offset = 2;
|
|
47
|
-
|
|
48
|
-
if (!masked) throw new Error('Client frames must be masked');
|
|
49
|
-
|
|
50
|
-
if (payloadLen === 126) {
|
|
51
|
-
if (buffer.length < 4) return null;
|
|
52
|
-
payloadLen = buffer.readUInt16BE(2);
|
|
53
|
-
offset = 4;
|
|
54
|
-
} else if (payloadLen === 127) {
|
|
55
|
-
if (buffer.length < 10) return null;
|
|
56
|
-
payloadLen = Number(buffer.readBigUInt64BE(2));
|
|
57
|
-
offset = 10;
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
const maskOffset = offset;
|
|
61
|
-
const dataOffset = offset + 4;
|
|
62
|
-
const totalLen = dataOffset + payloadLen;
|
|
63
|
-
if (buffer.length < totalLen) return null;
|
|
64
|
-
|
|
65
|
-
const mask = buffer.slice(maskOffset, dataOffset);
|
|
66
|
-
const data = Buffer.alloc(payloadLen);
|
|
67
|
-
for (let i = 0; i < payloadLen; i++) {
|
|
68
|
-
data[i] = buffer[dataOffset + i] ^ mask[i % 4];
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
return { opcode, payload: data, bytesConsumed: totalLen };
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
// ========== Configuration ==========
|
|
75
|
-
|
|
76
|
-
const PORT = process.env.BRAINSTORM_PORT || (49152 + Math.floor(Math.random() * 16383));
|
|
77
|
-
const HOST = process.env.BRAINSTORM_HOST || '127.0.0.1';
|
|
78
|
-
const URL_HOST = process.env.BRAINSTORM_URL_HOST || (HOST === '127.0.0.1' ? 'localhost' : HOST);
|
|
79
|
-
const SCREEN_DIR = process.env.BRAINSTORM_DIR || '/tmp/brainstorm';
|
|
80
|
-
const OWNER_PID = process.env.BRAINSTORM_OWNER_PID ? Number(process.env.BRAINSTORM_OWNER_PID) : null;
|
|
81
|
-
|
|
82
|
-
const MIME_TYPES = {
|
|
83
|
-
'.html': 'text/html', '.css': 'text/css', '.js': 'application/javascript',
|
|
84
|
-
'.json': 'application/json', '.png': 'image/png', '.jpg': 'image/jpeg',
|
|
85
|
-
'.jpeg': 'image/jpeg', '.gif': 'image/gif', '.svg': 'image/svg+xml'
|
|
86
|
-
};
|
|
87
|
-
|
|
88
|
-
// ========== Templates and Constants ==========
|
|
89
|
-
|
|
90
|
-
const WAITING_PAGE = `<!DOCTYPE html>
|
|
91
|
-
<html>
|
|
92
|
-
<head><meta charset="utf-8"><title>Brainstorm Companion</title>
|
|
93
|
-
<style>body { font-family: system-ui, sans-serif; padding: 2rem; max-width: 800px; margin: 0 auto; }
|
|
94
|
-
h1 { color: #333; } p { color: #666; }</style>
|
|
95
|
-
</head>
|
|
96
|
-
<body><h1>Brainstorm Companion</h1>
|
|
97
|
-
<p>Waiting for the agent to push a screen...</p></body></html>`;
|
|
98
|
-
|
|
99
|
-
const frameTemplate = fs.readFileSync(path.join(__dirname, 'frame-template.html'), 'utf-8');
|
|
100
|
-
const helperScript = fs.readFileSync(path.join(__dirname, 'helper.js'), 'utf-8');
|
|
101
|
-
const helperInjection = '<script>\n' + helperScript + '\n</script>';
|
|
102
|
-
|
|
103
|
-
// ========== Helper Functions ==========
|
|
104
|
-
|
|
105
|
-
function isFullDocument(html) {
|
|
106
|
-
const trimmed = html.trimStart().toLowerCase();
|
|
107
|
-
return trimmed.startsWith('<!doctype') || trimmed.startsWith('<html');
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
function wrapInFrame(content) {
|
|
111
|
-
return frameTemplate.replace('<!-- CONTENT -->', content);
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
function getNewestScreen() {
|
|
115
|
-
const files = fs.readdirSync(SCREEN_DIR)
|
|
116
|
-
.filter(f => f.endsWith('.html'))
|
|
117
|
-
.map(f => {
|
|
118
|
-
const fp = path.join(SCREEN_DIR, f);
|
|
119
|
-
return { path: fp, mtime: fs.statSync(fp).mtime.getTime() };
|
|
120
|
-
})
|
|
121
|
-
.sort((a, b) => b.mtime - a.mtime);
|
|
122
|
-
return files.length > 0 ? files[0].path : null;
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
// ========== HTTP Request Handler ==========
|
|
126
|
-
|
|
127
|
-
function handleRequest(req, res) {
|
|
128
|
-
touchActivity();
|
|
129
|
-
if (req.method === 'GET' && req.url === '/') {
|
|
130
|
-
const screenFile = getNewestScreen();
|
|
131
|
-
let html = screenFile
|
|
132
|
-
? (raw => isFullDocument(raw) ? raw : wrapInFrame(raw))(fs.readFileSync(screenFile, 'utf-8'))
|
|
133
|
-
: WAITING_PAGE;
|
|
134
|
-
|
|
135
|
-
if (html.includes('</body>')) {
|
|
136
|
-
html = html.replace('</body>', helperInjection + '\n</body>');
|
|
137
|
-
} else {
|
|
138
|
-
html += helperInjection;
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });
|
|
142
|
-
res.end(html);
|
|
143
|
-
} else if (req.method === 'GET' && req.url.startsWith('/files/')) {
|
|
144
|
-
const fileName = req.url.slice(7);
|
|
145
|
-
const filePath = path.join(SCREEN_DIR, path.basename(fileName));
|
|
146
|
-
if (!fs.existsSync(filePath)) {
|
|
147
|
-
res.writeHead(404);
|
|
148
|
-
res.end('Not found');
|
|
149
|
-
return;
|
|
150
|
-
}
|
|
151
|
-
const ext = path.extname(filePath).toLowerCase();
|
|
152
|
-
const contentType = MIME_TYPES[ext] || 'application/octet-stream';
|
|
153
|
-
res.writeHead(200, { 'Content-Type': contentType });
|
|
154
|
-
res.end(fs.readFileSync(filePath));
|
|
155
|
-
} else {
|
|
156
|
-
res.writeHead(404);
|
|
157
|
-
res.end('Not found');
|
|
158
|
-
}
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
// ========== WebSocket Connection Handling ==========
|
|
162
|
-
|
|
163
|
-
const clients = new Set();
|
|
164
|
-
|
|
165
|
-
function handleUpgrade(req, socket) {
|
|
166
|
-
const key = req.headers['sec-websocket-key'];
|
|
167
|
-
if (!key) { socket.destroy(); return; }
|
|
168
|
-
|
|
169
|
-
const accept = computeAcceptKey(key);
|
|
170
|
-
socket.write(
|
|
171
|
-
'HTTP/1.1 101 Switching Protocols\r\n' +
|
|
172
|
-
'Upgrade: websocket\r\n' +
|
|
173
|
-
'Connection: Upgrade\r\n' +
|
|
174
|
-
'Sec-WebSocket-Accept: ' + accept + '\r\n\r\n'
|
|
175
|
-
);
|
|
176
|
-
|
|
177
|
-
let buffer = Buffer.alloc(0);
|
|
178
|
-
clients.add(socket);
|
|
179
|
-
|
|
180
|
-
socket.on('data', (chunk) => {
|
|
181
|
-
buffer = Buffer.concat([buffer, chunk]);
|
|
182
|
-
while (buffer.length > 0) {
|
|
183
|
-
let result;
|
|
184
|
-
try {
|
|
185
|
-
result = decodeFrame(buffer);
|
|
186
|
-
} catch (e) {
|
|
187
|
-
socket.end(encodeFrame(OPCODES.CLOSE, Buffer.alloc(0)));
|
|
188
|
-
clients.delete(socket);
|
|
189
|
-
return;
|
|
190
|
-
}
|
|
191
|
-
if (!result) break;
|
|
192
|
-
buffer = buffer.slice(result.bytesConsumed);
|
|
193
|
-
|
|
194
|
-
switch (result.opcode) {
|
|
195
|
-
case OPCODES.TEXT:
|
|
196
|
-
handleMessage(result.payload.toString());
|
|
197
|
-
break;
|
|
198
|
-
case OPCODES.CLOSE:
|
|
199
|
-
socket.end(encodeFrame(OPCODES.CLOSE, Buffer.alloc(0)));
|
|
200
|
-
clients.delete(socket);
|
|
201
|
-
return;
|
|
202
|
-
case OPCODES.PING:
|
|
203
|
-
socket.write(encodeFrame(OPCODES.PONG, result.payload));
|
|
204
|
-
break;
|
|
205
|
-
case OPCODES.PONG:
|
|
206
|
-
break;
|
|
207
|
-
default: {
|
|
208
|
-
const closeBuf = Buffer.alloc(2);
|
|
209
|
-
closeBuf.writeUInt16BE(1003);
|
|
210
|
-
socket.end(encodeFrame(OPCODES.CLOSE, closeBuf));
|
|
211
|
-
clients.delete(socket);
|
|
212
|
-
return;
|
|
213
|
-
}
|
|
214
|
-
}
|
|
215
|
-
}
|
|
216
|
-
});
|
|
217
|
-
|
|
218
|
-
socket.on('close', () => clients.delete(socket));
|
|
219
|
-
socket.on('error', () => clients.delete(socket));
|
|
220
|
-
}
|
|
221
|
-
|
|
222
|
-
function handleMessage(text) {
|
|
223
|
-
let event;
|
|
224
|
-
try {
|
|
225
|
-
event = JSON.parse(text);
|
|
226
|
-
} catch (e) {
|
|
227
|
-
console.error('Failed to parse WebSocket message:', e.message);
|
|
228
|
-
return;
|
|
229
|
-
}
|
|
230
|
-
touchActivity();
|
|
231
|
-
console.log(JSON.stringify({ source: 'user-event', ...event }));
|
|
232
|
-
if (event.choice) {
|
|
233
|
-
const eventsFile = path.join(SCREEN_DIR, '.events');
|
|
234
|
-
fs.appendFileSync(eventsFile, JSON.stringify(event) + '\n');
|
|
235
|
-
}
|
|
236
|
-
}
|
|
237
|
-
|
|
238
|
-
function broadcast(msg) {
|
|
239
|
-
const frame = encodeFrame(OPCODES.TEXT, Buffer.from(JSON.stringify(msg)));
|
|
240
|
-
for (const socket of clients) {
|
|
241
|
-
try { socket.write(frame); } catch (e) { clients.delete(socket); }
|
|
242
|
-
}
|
|
243
|
-
}
|
|
244
|
-
|
|
245
|
-
// ========== Activity Tracking ==========
|
|
246
|
-
|
|
247
|
-
const IDLE_TIMEOUT_MS = 30 * 60 * 1000; // 30 minutes
|
|
248
|
-
let lastActivity = Date.now();
|
|
249
|
-
|
|
250
|
-
function touchActivity() {
|
|
251
|
-
lastActivity = Date.now();
|
|
252
|
-
}
|
|
253
|
-
|
|
254
|
-
// ========== File Watching ==========
|
|
255
|
-
|
|
256
|
-
const debounceTimers = new Map();
|
|
257
|
-
|
|
258
|
-
// ========== Server Startup ==========
|
|
259
|
-
|
|
260
|
-
function startServer() {
|
|
261
|
-
if (!fs.existsSync(SCREEN_DIR)) fs.mkdirSync(SCREEN_DIR, { recursive: true });
|
|
262
|
-
|
|
263
|
-
// Track known files to distinguish new screens from updates.
|
|
264
|
-
// macOS fs.watch reports 'rename' for both new files and overwrites,
|
|
265
|
-
// so we can't rely on eventType alone.
|
|
266
|
-
const knownFiles = new Set(
|
|
267
|
-
fs.readdirSync(SCREEN_DIR).filter(f => f.endsWith('.html'))
|
|
268
|
-
);
|
|
269
|
-
|
|
270
|
-
const server = http.createServer(handleRequest);
|
|
271
|
-
server.on('upgrade', handleUpgrade);
|
|
272
|
-
|
|
273
|
-
const watcher = fs.watch(SCREEN_DIR, (eventType, filename) => {
|
|
274
|
-
if (!filename || !filename.endsWith('.html')) return;
|
|
275
|
-
|
|
276
|
-
if (debounceTimers.has(filename)) clearTimeout(debounceTimers.get(filename));
|
|
277
|
-
debounceTimers.set(filename, setTimeout(() => {
|
|
278
|
-
debounceTimers.delete(filename);
|
|
279
|
-
const filePath = path.join(SCREEN_DIR, filename);
|
|
280
|
-
|
|
281
|
-
if (!fs.existsSync(filePath)) return; // file was deleted
|
|
282
|
-
touchActivity();
|
|
283
|
-
|
|
284
|
-
if (!knownFiles.has(filename)) {
|
|
285
|
-
knownFiles.add(filename);
|
|
286
|
-
const eventsFile = path.join(SCREEN_DIR, '.events');
|
|
287
|
-
if (fs.existsSync(eventsFile)) fs.unlinkSync(eventsFile);
|
|
288
|
-
console.log(JSON.stringify({ type: 'screen-added', file: filePath }));
|
|
289
|
-
} else {
|
|
290
|
-
console.log(JSON.stringify({ type: 'screen-updated', file: filePath }));
|
|
291
|
-
}
|
|
292
|
-
|
|
293
|
-
broadcast({ type: 'reload' });
|
|
294
|
-
}, 100));
|
|
295
|
-
});
|
|
296
|
-
watcher.on('error', (err) => console.error('fs.watch error:', err.message));
|
|
297
|
-
|
|
298
|
-
function shutdown(reason) {
|
|
299
|
-
console.log(JSON.stringify({ type: 'server-stopped', reason }));
|
|
300
|
-
const infoFile = path.join(SCREEN_DIR, '.server-info');
|
|
301
|
-
if (fs.existsSync(infoFile)) fs.unlinkSync(infoFile);
|
|
302
|
-
fs.writeFileSync(
|
|
303
|
-
path.join(SCREEN_DIR, '.server-stopped'),
|
|
304
|
-
JSON.stringify({ reason, timestamp: Date.now() }) + '\n'
|
|
305
|
-
);
|
|
306
|
-
watcher.close();
|
|
307
|
-
clearInterval(lifecycleCheck);
|
|
308
|
-
server.close(() => process.exit(0));
|
|
309
|
-
}
|
|
310
|
-
|
|
311
|
-
function ownerAlive() {
|
|
312
|
-
if (!OWNER_PID) return true;
|
|
313
|
-
try { process.kill(OWNER_PID, 0); return true; } catch (e) { return false; }
|
|
314
|
-
}
|
|
315
|
-
|
|
316
|
-
// Check every 60s: exit if owner process died or idle for 30 minutes
|
|
317
|
-
const lifecycleCheck = setInterval(() => {
|
|
318
|
-
if (!ownerAlive()) shutdown('owner process exited');
|
|
319
|
-
else if (Date.now() - lastActivity > IDLE_TIMEOUT_MS) shutdown('idle timeout');
|
|
320
|
-
}, 60 * 1000);
|
|
321
|
-
lifecycleCheck.unref();
|
|
322
|
-
|
|
323
|
-
server.listen(PORT, HOST, () => {
|
|
324
|
-
const info = JSON.stringify({
|
|
325
|
-
type: 'server-started', port: Number(PORT), host: HOST,
|
|
326
|
-
url_host: URL_HOST, url: 'http://' + URL_HOST + ':' + PORT,
|
|
327
|
-
screen_dir: SCREEN_DIR
|
|
328
|
-
});
|
|
329
|
-
console.log(info);
|
|
330
|
-
fs.writeFileSync(path.join(SCREEN_DIR, '.server-info'), info + '\n');
|
|
331
|
-
});
|
|
332
|
-
}
|
|
333
|
-
|
|
334
|
-
if (require.main === module) {
|
|
335
|
-
startServer();
|
|
336
|
-
}
|
|
337
|
-
|
|
338
|
-
module.exports = { computeAcceptKey, encodeFrame, decodeFrame, OPCODES };
|
|
@@ -1,153 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env bash
|
|
2
|
-
# Start the brainstorm server and output connection info
|
|
3
|
-
# Usage: start-server.sh [--project-dir <path>] [--host <bind-host>] [--url-host <display-host>] [--foreground] [--background]
|
|
4
|
-
#
|
|
5
|
-
# Starts server on a random high port, outputs JSON with URL.
|
|
6
|
-
# Each session gets its own directory to avoid conflicts.
|
|
7
|
-
#
|
|
8
|
-
# Options:
|
|
9
|
-
# --project-dir <path> Store session files under <path>/.superpowers/brainstorm/
|
|
10
|
-
# instead of /tmp. Files persist after server stops.
|
|
11
|
-
# --host <bind-host> Host/interface to bind (default: 127.0.0.1).
|
|
12
|
-
# Use 0.0.0.0 in remote/containerized environments.
|
|
13
|
-
# --url-host <host> Hostname shown in returned URL JSON.
|
|
14
|
-
# --foreground Run server in the current terminal (no backgrounding).
|
|
15
|
-
# --background Force background mode (overrides Codex auto-foreground).
|
|
16
|
-
|
|
17
|
-
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
|
|
18
|
-
|
|
19
|
-
# Parse arguments
|
|
20
|
-
PROJECT_DIR=""
|
|
21
|
-
FOREGROUND="false"
|
|
22
|
-
FORCE_BACKGROUND="false"
|
|
23
|
-
BIND_HOST="127.0.0.1"
|
|
24
|
-
URL_HOST=""
|
|
25
|
-
while [[ $# -gt 0 ]]; do
|
|
26
|
-
case "$1" in
|
|
27
|
-
--project-dir)
|
|
28
|
-
PROJECT_DIR="$2"
|
|
29
|
-
shift 2
|
|
30
|
-
;;
|
|
31
|
-
--host)
|
|
32
|
-
BIND_HOST="$2"
|
|
33
|
-
shift 2
|
|
34
|
-
;;
|
|
35
|
-
--url-host)
|
|
36
|
-
URL_HOST="$2"
|
|
37
|
-
shift 2
|
|
38
|
-
;;
|
|
39
|
-
--foreground|--no-daemon)
|
|
40
|
-
FOREGROUND="true"
|
|
41
|
-
shift
|
|
42
|
-
;;
|
|
43
|
-
--background|--daemon)
|
|
44
|
-
FORCE_BACKGROUND="true"
|
|
45
|
-
shift
|
|
46
|
-
;;
|
|
47
|
-
*)
|
|
48
|
-
echo "{\"error\": \"Unknown argument: $1\"}"
|
|
49
|
-
exit 1
|
|
50
|
-
;;
|
|
51
|
-
esac
|
|
52
|
-
done
|
|
53
|
-
|
|
54
|
-
if [[ -z "$URL_HOST" ]]; then
|
|
55
|
-
if [[ "$BIND_HOST" == "127.0.0.1" || "$BIND_HOST" == "localhost" ]]; then
|
|
56
|
-
URL_HOST="localhost"
|
|
57
|
-
else
|
|
58
|
-
URL_HOST="$BIND_HOST"
|
|
59
|
-
fi
|
|
60
|
-
fi
|
|
61
|
-
|
|
62
|
-
# Some environments reap detached/background processes. Auto-foreground when detected.
|
|
63
|
-
if [[ -n "${CODEX_CI:-}" && "$FOREGROUND" != "true" && "$FORCE_BACKGROUND" != "true" ]]; then
|
|
64
|
-
FOREGROUND="true"
|
|
65
|
-
fi
|
|
66
|
-
|
|
67
|
-
# Windows/Git Bash reaps nohup background processes. Auto-foreground when detected.
|
|
68
|
-
if [[ "$FOREGROUND" != "true" && "$FORCE_BACKGROUND" != "true" ]]; then
|
|
69
|
-
case "${OSTYPE:-}" in
|
|
70
|
-
msys*|cygwin*|mingw*) FOREGROUND="true" ;;
|
|
71
|
-
esac
|
|
72
|
-
if [[ -n "${MSYSTEM:-}" ]]; then
|
|
73
|
-
FOREGROUND="true"
|
|
74
|
-
fi
|
|
75
|
-
fi
|
|
76
|
-
|
|
77
|
-
# Generate unique session directory
|
|
78
|
-
SESSION_ID="$$-$(date +%s)"
|
|
79
|
-
|
|
80
|
-
if [[ -n "$PROJECT_DIR" ]]; then
|
|
81
|
-
SCREEN_DIR="${PROJECT_DIR}/.superpowers/brainstorm/${SESSION_ID}"
|
|
82
|
-
else
|
|
83
|
-
SCREEN_DIR="/tmp/brainstorm-${SESSION_ID}"
|
|
84
|
-
fi
|
|
85
|
-
|
|
86
|
-
PID_FILE="${SCREEN_DIR}/.server.pid"
|
|
87
|
-
LOG_FILE="${SCREEN_DIR}/.server.log"
|
|
88
|
-
|
|
89
|
-
# Create fresh session directory
|
|
90
|
-
mkdir -p "$SCREEN_DIR"
|
|
91
|
-
|
|
92
|
-
# Kill any existing server
|
|
93
|
-
if [[ -f "$PID_FILE" ]]; then
|
|
94
|
-
old_pid=$(cat "$PID_FILE")
|
|
95
|
-
kill "$old_pid" 2>/dev/null
|
|
96
|
-
rm -f "$PID_FILE"
|
|
97
|
-
fi
|
|
98
|
-
|
|
99
|
-
cd "$SCRIPT_DIR"
|
|
100
|
-
|
|
101
|
-
# Resolve the harness PID (grandparent of this script).
|
|
102
|
-
# $PPID is the ephemeral shell the harness spawned to run us — it dies
|
|
103
|
-
# when this script exits. The harness itself is $PPID's parent.
|
|
104
|
-
OWNER_PID="$(ps -o ppid= -p "$PPID" 2>/dev/null | tr -d ' ')"
|
|
105
|
-
if [[ -z "$OWNER_PID" || "$OWNER_PID" == "1" ]]; then
|
|
106
|
-
OWNER_PID="$PPID"
|
|
107
|
-
fi
|
|
108
|
-
|
|
109
|
-
# On Windows/MSYS2, the MSYS2 PID namespace is invisible to Node.js.
|
|
110
|
-
# Skip owner-PID monitoring — the 30-minute idle timeout prevents orphans.
|
|
111
|
-
case "${OSTYPE:-}" in
|
|
112
|
-
msys*|cygwin*|mingw*) OWNER_PID="" ;;
|
|
113
|
-
esac
|
|
114
|
-
|
|
115
|
-
# Foreground mode for environments that reap detached/background processes.
|
|
116
|
-
if [[ "$FOREGROUND" == "true" ]]; then
|
|
117
|
-
echo "$$" > "$PID_FILE"
|
|
118
|
-
env BRAINSTORM_DIR="$SCREEN_DIR" BRAINSTORM_HOST="$BIND_HOST" BRAINSTORM_URL_HOST="$URL_HOST" BRAINSTORM_OWNER_PID="$OWNER_PID" node server.cjs
|
|
119
|
-
exit $?
|
|
120
|
-
fi
|
|
121
|
-
|
|
122
|
-
# Start server, capturing output to log file
|
|
123
|
-
# Use nohup to survive shell exit; disown to remove from job table
|
|
124
|
-
nohup env BRAINSTORM_DIR="$SCREEN_DIR" BRAINSTORM_HOST="$BIND_HOST" BRAINSTORM_URL_HOST="$URL_HOST" BRAINSTORM_OWNER_PID="$OWNER_PID" node server.cjs > "$LOG_FILE" 2>&1 &
|
|
125
|
-
SERVER_PID=$!
|
|
126
|
-
disown "$SERVER_PID" 2>/dev/null
|
|
127
|
-
echo "$SERVER_PID" > "$PID_FILE"
|
|
128
|
-
|
|
129
|
-
# Wait for server-started message (check log file)
|
|
130
|
-
for i in {1..50}; do
|
|
131
|
-
if grep -q "server-started" "$LOG_FILE" 2>/dev/null; then
|
|
132
|
-
# Verify server is still alive after a short window (catches process reapers)
|
|
133
|
-
alive="true"
|
|
134
|
-
for _ in {1..20}; do
|
|
135
|
-
if ! kill -0 "$SERVER_PID" 2>/dev/null; then
|
|
136
|
-
alive="false"
|
|
137
|
-
break
|
|
138
|
-
fi
|
|
139
|
-
sleep 0.1
|
|
140
|
-
done
|
|
141
|
-
if [[ "$alive" != "true" ]]; then
|
|
142
|
-
echo "{\"error\": \"Server started but was killed. Retry in a persistent terminal with: $SCRIPT_DIR/start-server.sh${PROJECT_DIR:+ --project-dir $PROJECT_DIR} --host $BIND_HOST --url-host $URL_HOST --foreground\"}"
|
|
143
|
-
exit 1
|
|
144
|
-
fi
|
|
145
|
-
grep "server-started" "$LOG_FILE" | head -1
|
|
146
|
-
exit 0
|
|
147
|
-
fi
|
|
148
|
-
sleep 0.1
|
|
149
|
-
done
|
|
150
|
-
|
|
151
|
-
# Timeout - server didn't start
|
|
152
|
-
echo '{"error": "Server failed to start within 5 seconds"}'
|
|
153
|
-
exit 1
|
|
@@ -1,55 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env bash
|
|
2
|
-
# Stop the brainstorm server and clean up
|
|
3
|
-
# Usage: stop-server.sh <screen_dir>
|
|
4
|
-
#
|
|
5
|
-
# Kills the server process. Only deletes session directory if it's
|
|
6
|
-
# under /tmp (ephemeral). Persistent directories (.superpowers/) are
|
|
7
|
-
# kept so mockups can be reviewed later.
|
|
8
|
-
|
|
9
|
-
SCREEN_DIR="$1"
|
|
10
|
-
|
|
11
|
-
if [[ -z "$SCREEN_DIR" ]]; then
|
|
12
|
-
echo '{"error": "Usage: stop-server.sh <screen_dir>"}'
|
|
13
|
-
exit 1
|
|
14
|
-
fi
|
|
15
|
-
|
|
16
|
-
PID_FILE="${SCREEN_DIR}/.server.pid"
|
|
17
|
-
|
|
18
|
-
if [[ -f "$PID_FILE" ]]; then
|
|
19
|
-
pid=$(cat "$PID_FILE")
|
|
20
|
-
|
|
21
|
-
# Try to stop gracefully, fallback to force if still alive
|
|
22
|
-
kill "$pid" 2>/dev/null || true
|
|
23
|
-
|
|
24
|
-
# Wait for graceful shutdown (up to ~2s)
|
|
25
|
-
for i in {1..20}; do
|
|
26
|
-
if ! kill -0 "$pid" 2>/dev/null; then
|
|
27
|
-
break
|
|
28
|
-
fi
|
|
29
|
-
sleep 0.1
|
|
30
|
-
done
|
|
31
|
-
|
|
32
|
-
# If still running, escalate to SIGKILL
|
|
33
|
-
if kill -0 "$pid" 2>/dev/null; then
|
|
34
|
-
kill -9 "$pid" 2>/dev/null || true
|
|
35
|
-
|
|
36
|
-
# Give SIGKILL a moment to take effect
|
|
37
|
-
sleep 0.1
|
|
38
|
-
fi
|
|
39
|
-
|
|
40
|
-
if kill -0 "$pid" 2>/dev/null; then
|
|
41
|
-
echo '{"status": "failed", "error": "process still running"}'
|
|
42
|
-
exit 1
|
|
43
|
-
fi
|
|
44
|
-
|
|
45
|
-
rm -f "$PID_FILE" "${SCREEN_DIR}/.server.log"
|
|
46
|
-
|
|
47
|
-
# Only delete ephemeral /tmp directories
|
|
48
|
-
if [[ "$SCREEN_DIR" == /tmp/* ]]; then
|
|
49
|
-
rm -rf "$SCREEN_DIR"
|
|
50
|
-
fi
|
|
51
|
-
|
|
52
|
-
echo '{"status": "stopped"}'
|
|
53
|
-
else
|
|
54
|
-
echo '{"status": "not_running"}'
|
|
55
|
-
fi
|
|
@@ -1,48 +0,0 @@
|
|
|
1
|
-
# 规格文档审查员提示模板
|
|
2
|
-
|
|
3
|
-
调度规格文档审查员子代理时使用此模板。
|
|
4
|
-
|
|
5
|
-
**用途:** 验证规格是否完整、一致,并为实现计划做好准备。
|
|
6
|
-
|
|
7
|
-
**调度时机:** 规格文档写入 docs/superpowers/specs/ 之后
|
|
8
|
-
|
|
9
|
-
```
|
|
10
|
-
Task tool(通用):
|
|
11
|
-
description: "审查规格文档"
|
|
12
|
-
prompt: |
|
|
13
|
-
你是一名规格文档审查员。验证此规格是否完整并准备好进行计划编写。
|
|
14
|
-
|
|
15
|
-
**待审查规格:** [SPEC_FILE_PATH]
|
|
16
|
-
|
|
17
|
-
## 检查内容
|
|
18
|
-
|
|
19
|
-
| 类别 | 检查要点 |
|
|
20
|
-
|------|----------|
|
|
21
|
-
| 完整性 | TODO、占位符、"TBD"、不完整的章节 |
|
|
22
|
-
| 一致性 | 内部矛盾、相互冲突的需求 |
|
|
23
|
-
| 清晰度 | 需求模糊到可能导致构建出错误的东西 |
|
|
24
|
-
| 范围 | 是否足够聚焦以用于单个计划——而非涵盖多个独立子系统 |
|
|
25
|
-
| YAGNI | 未请求的功能、过度设计 |
|
|
26
|
-
|
|
27
|
-
## 校准标准
|
|
28
|
-
|
|
29
|
-
**只标记会在实现计划阶段造成实际问题的事项。**
|
|
30
|
-
缺失的章节、矛盾之处、或者模糊到可能被两种不同方式理解的需求——
|
|
31
|
-
这些才是问题。措辞上的小改进、风格偏好、以及"某些章节不如其他章节详细"则不是。
|
|
32
|
-
|
|
33
|
-
除非存在会导致计划出错的严重缺陷,否则应予以通过。
|
|
34
|
-
|
|
35
|
-
## 输出格式
|
|
36
|
-
|
|
37
|
-
## 规格审查
|
|
38
|
-
|
|
39
|
-
**状态:** 通过 | 发现问题
|
|
40
|
-
|
|
41
|
-
**问题(如有):**
|
|
42
|
-
- [章节 X]:[具体问题] - [为什么这对计划编写很重要]
|
|
43
|
-
|
|
44
|
-
**建议(仅供参考,不阻止通过):**
|
|
45
|
-
- [改进建议]
|
|
46
|
-
```
|
|
47
|
-
|
|
48
|
-
**审查员返回:** 状态、问题(如有)、建议
|