Skip to content

可视化伴侣认证加固实施计划

由 Markdown 原样翻译并转换为 Astro Starlight MDX 格式。

对于 agentic workers: REQUIRED SUB-SKILL: 使用 superpowers:subagent-driven-development (推荐) or superpowers:executing-plans to implement this 计划 task-by-task. Steps use checkbox (- [ ]) syntax for tracking.

Goal: Harden the brainstorming visual companion auth and reconnect flow while preserving trusted same-origin screen JavaScript and future vendored UI libraries.

架构: Keyed root 加载 become a bootstrap step that sets the cookie, stores the key in tab-scoped sessionStorage, and navigates to a bare / screen URL. WebSockets require valid auth plus browser same-origin Origin, while /files/* uses realpath containment to prevent content-directory escapes.

Tech Stack: Node.js built-ins (http, fs, path, crypto), zero 运行时 依赖, 现有 ws test 依赖, Bash start/stop scripts, repo shell lint script.

Important: Do not commit during execution unless Drew explicitly asks. This repository’s instructions override the generic 计划 模板’s commit cadence.


  • 修改: skills/brainstorming/scripts/server.cjs
    • 添加 bootstrap response.
    • 添加 shared 安全 headers.
    • 添加 WebSocket Origin validation.
    • 添加 /files/* realpath containment.
  • 修改: skills/brainstorming/scripts/helper.js
    • Read the stored session key and append it to the WebSocket URL.
  • 修改: tests/brainstorm-server/auth.test.js
    • 添加 bootstrap, header, same-origin WS, cross-origin WS, and cookie/file auth regressions.
  • 修改: tests/brainstorm-server/helper.test.js
    • 添加 mocked-browser coverage for sessionStorage-backed WS URLs.
  • 修改: tests/brainstorm-server/server.test.js
    • 添加 symlink containment regression for /files/*.
  • 修改: tests/brainstorm-server/lifecycle.test.js
    • Make the start-server timeout flag test force 背景 mode.
    • 添加 重启 reconnect credential coverage if it fits the 现有 lifecycle helper.
  • 修改: skills/brainstorming/scripts/start-server.sh
    • Fix shell lint.
  • 修改: skills/brainstorming/scripts/stop-server.sh
    • Fix shell lint.
  • 修改: .gitignore
    • 添加 .superpowers/.
  • Optional docs update: skills/brainstorming/visual-companion.md
    • Mention bootstrap URL stripping and trusted same-origin screen JS if the code behavior changes need operator-facing explanation.

文件:

  • 修改: tests/brainstorm-server/auth.test.js

  • 修改: skills/brainstorming/scripts/server.cjs

  • 步骤 1: 添加 RED tests for bootstrap behavior

In tests/brainstorm-server/auth.test.js, add tests after the 现有 valid-key root test:

await test('GET / with valid query returns bootstrap instead of screen content', async () => {
const res = await get('/', { key: TOKEN });
assert.strictEqual(res.status, 200);
assert(res.body.includes('sessionStorage'), 'bootstrap should store the session key in tab storage');
assert(res.body.includes('location.replace'), 'bootstrap should navigate to the bare root URL');
assert(!res.body.includes('Secret screen'), 'bootstrap must not serve screen HTML at the keyed URL');
});
await test('GET / with valid cookie serves the screen after bootstrap', async () => {
const res = await get('/', { cookie: `${COOKIE_NAME}=${TOKEN}` });
assert.strictEqual(res.status, 200);
assert(res.body.includes('Secret screen'), 'cookie-authenticated bare root should serve the screen');
assert(!res.body.includes('sessionStorage'), 'bare screen response should not be the bootstrap page');
});

Keep the 现有 cookie test if present; merge assertions rather than duplicating the same test name.

  • 步骤 2: 验证 RED

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node auth.test.js

预期: the 新 bootstrap test fails because 当前 GET /?key=... serves Secret screen directly and does not include the bootstrap sessionStorage/location.replace code.

  • 步骤 3: Implement minimal bootstrap response

In skills/brainstorming/scripts/server.cjs, add a helper near the page constants:

function bootstrapPage(key) {
const jsonKey = JSON.stringify(String(key));
return `<!DOCTYPE html>
<html>
<head><meta charset="utf-8"><title>Opening Brainstorm Companion</title></head>
<body>
<script>
sessionStorage.setItem('brainstorm-session-key', ${jsonKey});
location.replace('/');
</script>
</body>
</html>`;
}

然后 in handleRequest, after 授权 and cookie setting but before serving screen HTML, 检测 a valid query key on root:

function queryKey(url) {
const q = url.indexOf('?');
if (q < 0) return null;
return new URLSearchParams(url.slice(q + 1)).get('key');
}

使用 it in handleRequest:

const pathname = pathnameOf(req.url);
const keyFromQuery = queryKey(req.url);
if (req.method === 'GET' && pathname === '/' && keyFromQuery && timingSafeEqualStr(keyFromQuery, TOKEN)) {
res.writeHead(200, securityHeaders({ 'Content-Type': 'text/html; charset=utf-8' }));
res.end(bootstrapPage(keyFromQuery));
return;
}

This assumes Task 4 will introduce securityHeaders. 如果 implementing Task 1 first, temporarily use:

res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });

and replace it in Task 4.

  • 步骤 4: 验证 GREEN

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node auth.test.js

预期: all auth tests pass, including the 新 bootstrap tests.

文件:

  • 修改: tests/brainstorm-server/auth.test.js

  • 修改: skills/brainstorming/scripts/server.cjs

  • 步骤 1: 添加 RED tests for same-origin and cross-origin WS

In tests/brainstorm-server/auth.test.js, extend wsConnect to accept an origin option:

function wsConnect({ key, cookie, origin } = {}) {
const url = `ws://localhost:${TEST_PORT}/` + (key !== undefined ? `?key=${key}` : '');
const headers = {};
if (cookie) headers['Cookie'] = cookie;
if (origin) headers['Origin'] = origin;
const ws = new WebSocket(url, Object.keys(headers).length ? { headers } : {});
return new Promise((resolve) => {
let settled = false;
const done = (outcome) => { if (!settled) { settled = true; resolve({ outcome, ws }); } };
ws.on('open', () => done('opened'));
ws.on('error', () => done('rejected'));
ws.on('close', () => done('rejected'));
setTimeout(() => done('rejected'), 1500);
});
}

然后 add:

await test('WS upgrade with valid cookie and same-origin Origin opens', async () => {
const { outcome, ws } = await wsConnect({
cookie: `${COOKIE_NAME}=${TOKEN}`,
origin: `http://localhost:${TEST_PORT}`
});
ws.close();
assert.strictEqual(outcome, 'opened');
});
await test('WS upgrade with valid cookie but cross-origin Origin is rejected', async () => {
const eventsFile = path.join(TEST_DIR, 'state', 'events');
if (fs.existsSync(eventsFile)) fs.unlinkSync(eventsFile);
const { outcome, ws } = await wsConnect({
cookie: `${COOKIE_NAME}=${TOKEN}`,
origin: 'http://localhost:9999'
});
if (outcome === 'opened') {
ws.send(JSON.stringify({ type: 'choice', choice: 'attacker-injected', text: 'local attacker probe' }));
await sleep(300);
}
ws.close();
assert.strictEqual(outcome, 'rejected', 'cross-origin browser WS must not open even with cookie');
assert(!fs.existsSync(eventsFile), 'cross-origin WS must not write state/events');
});
  • 步骤 2: 验证 RED

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node auth.test.js

预期: cross-origin cookie WS test fails because 当前 服务器 accepts any cookie-authenticated WS regardless of Origin.

  • 步骤 3: Implement Origin check

In skills/brainstorming/scripts/server.cjs, add:

function isAllowedWebSocketOrigin(req) {
const origin = req.headers.origin;
if (!origin) return true; // non-browser clients still need the session key
const host = req.headers.host;
if (!host) return false;
return origin === 'http://' + host;
}

然后 update handleUpgrade:

function handleUpgrade(req, socket) {
if (!isAuthorized(req) || !isAllowedWebSocketOrigin(req)) { socket.destroy(); return; }
  • 步骤 4: 验证 GREEN

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node auth.test.js

预期: auth tests pass; cross-origin WS is rejected; same-origin and direct key WS still open.

Task 3: Helper Uses Stored Key 对于 Reconnect

Section titled “Task 3: Helper Uses Stored Key 对于 Reconnect”

文件:

  • 修改: tests/brainstorm-server/helper.test.js

  • 修改: skills/brainstorming/scripts/helper.js

  • 步骤 1: 添加 RED test for WebSocket URL key

In tests/brainstorm-server/helper.test.js, add a mocked-browser test near the reconnect state-machine tests:

test('uses sessionStorage key in the WebSocket URL when present', () => {
const e = makeEnv();
e.state.sessionKey = 'stored-key-abc';
e.boot();
assert.strictEqual(e.sockets[0].url, 'ws://localhost:7777/?key=stored-key-abc');
});

更新 makeEnv() so the returned object exposes sockets, and the mock window includes sessionStorage:

window: {
location: { host: 'localhost:7777', reload() { state.reloads++; } },
sessionStorage: { getItem: (key) => key === 'brainstorm-session-key' ? state.sessionKey : null }
},

Also add a 回退 test:

test('uses cookie-only WebSocket URL when no sessionStorage key is present', () => {
const e = makeEnv();
e.state.sessionKey = null;
e.boot();
assert.strictEqual(e.sockets[0].url, 'ws://localhost:7777');
});
  • 步骤 2: 验证 RED

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node helper.test.js

预期: stored-key test fails because 当前 helper uses ws://localhost:7777.

  • 步骤 3: Implement stored-key WS URL

In skills/brainstorming/scripts/helper.js, replace:

const WS_URL = 'ws://' + window.location.host;

使用:

function websocketUrl() {
let key = null;
try { key = window.sessionStorage && window.sessionStorage.getItem('brainstorm-session-key'); } catch (e) {}
return 'ws://' + window.location.host + (key ? '/?key=' + encodeURIComponent(key) : '');
}

然后 replace:

ws = new WebSocket(WS_URL);

使用:

ws = new WebSocket(websocketUrl());
  • 步骤 4: 验证 GREEN

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node helper.test.js

预期: helper tests pass.

文件:

  • 修改: tests/brainstorm-server/auth.test.js

  • 修改: skills/brainstorming/scripts/server.cjs

  • 步骤 1: 添加 RED header tests

In tests/brainstorm-server/auth.test.js, add:

await test('HTML responses include leak-reduction and anti-framing headers', async () => {
const res = await get('/', { key: TOKEN });
assert.strictEqual(res.headers['referrer-policy'], 'no-referrer');
assert.strictEqual(res.headers['cache-control'], 'no-store');
assert.strictEqual(res.headers['x-frame-options'], 'DENY');
assert.strictEqual(res.headers['content-security-policy'], "frame-ancestors 'none'");
assert.strictEqual(res.headers['cross-origin-resource-policy'], 'same-origin');
});
await test('403 responses include leak-reduction and anti-framing headers', async () => {
const res = await get('/');
assert.strictEqual(res.status, 403);
assert.strictEqual(res.headers['referrer-policy'], 'no-referrer');
assert.strictEqual(res.headers['cache-control'], 'no-store');
assert.strictEqual(res.headers['x-frame-options'], 'DENY');
assert.strictEqual(res.headers['content-security-policy'], "frame-ancestors 'none'");
assert.strictEqual(res.headers['cross-origin-resource-policy'], 'same-origin');
});
  • 步骤 2: 验证 RED

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node auth.test.js

预期: header tests fail because 当前 responses do not include these headers.

  • 步骤 3: Implement shared header helper

In skills/brainstorming/scripts/server.cjs, add:

function securityHeaders(headers = {}) {
return {
'Referrer-Policy': 'no-referrer',
'Cache-Control': 'no-store',
'X-Frame-Options': 'DENY',
'Content-Security-Policy': "frame-ancestors 'none'",
'Cross-Origin-Resource-Policy': 'same-origin',
...headers
};
}

更新 response writes in handleRequest:

res.writeHead(403, securityHeaders({ 'Content-Type': 'text/html; charset=utf-8' }));
res.writeHead(200, securityHeaders({ 'Content-Type': 'text/html; charset=utf-8' }));
res.writeHead(200, securityHeaders({ 'Content-Type': contentType }));

对于 404s:

res.writeHead(404, securityHeaders());
  • 步骤 4: 验证 GREEN

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node auth.test.js

预期: auth tests pass and header assertions are green.

文件:

  • 修改: tests/brainstorm-server/server.test.js

  • 修改: skills/brainstorming/scripts/server.cjs

  • 步骤 1: 添加 RED symlink escape test

In tests/brainstorm-server/server.test.js, after the /files/ empty-name test, add:

await test('does not serve symlinks that escape content dir via /files/', async () => {
const target = path.join(STATE_DIR, 'server-info');
const link = path.join(CONTENT_DIR, 'linked-server-info.txt');
try { fs.unlinkSync(link); } catch (e) {}
fs.symlinkSync(target, link);
const res = await fetch(`http://localhost:${TEST_PORT}/files/linked-server-info.txt`);
assert.strictEqual(res.status, 404, 'symlink to state/server-info must not be served');
assert(!res.body.includes('server-started'), 'response must not include server-info body');
});
  • 步骤 2: 验证 RED

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node server.test.js

预期: symlink test fails because 当前 /files/* follows symlinks and serves server-info.

  • 步骤 3: Implement containment helper

In skills/brainstorming/scripts/server.cjs, add:

function isRegularFileInsideContentDir(filePath) {
let stat, realContentDir, realFilePath;
try {
stat = fs.lstatSync(filePath);
if (stat.isSymbolicLink()) return false;
if (!stat.isFile()) return false;
realContentDir = fs.realpathSync(CONTENT_DIR);
realFilePath = fs.realpathSync(filePath);
} catch (e) {
return false;
}
return realFilePath.startsWith(realContentDir + path.sep);
}

替换 the /files/* guard with:

if (!fileName || fileName.startsWith('.') || !isRegularFileInsideContentDir(filePath)) {
res.writeHead(404, securityHeaders());
res.end('Not found');
return;
}
  • 步骤 4: 验证 GREEN

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node server.test.js

预期: 服务器 tests pass, including symlink rejection.

文件:

  • 修改: tests/brainstorm-server/lifecycle.test.js

  • 修改: skills/brainstorming/scripts/server.cjs

  • 修改: skills/brainstorming/scripts/helper.js

  • 步骤 1: 添加 RED integration test for same key over WS after 重启

In tests/brainstorm-server/lifecycle.test.js, add a test after the port/token persistence test:

await test('stored key can authenticate WebSocket after same-port restart', async () => {
const dir = fs.mkdtempSync('/tmp/bs-reconnect-');
const portFile = path.join(dir, '.last-port');
const tokenFile = path.join(dir, '.last-token');
const env = { ...process.env, BRAINSTORM_PORT_FILE: portFile, BRAINSTORM_TOKEN_FILE: tokenFile, BRAINSTORM_LIFECYCLE_CHECK_MS: 100000 };
const a = spawn('node', [SERVER], { env: { ...env, BRAINSTORM_DIR: path.join(dir, 's1') } });
let outA = ''; a.stdout.on('data', d => outA += d.toString());
for (let i = 0; i < 60 && !outA.includes('server-started'); i++) await sleep(50);
const infoA = firstServerStarted(outA);
const keyA = new URL(infoA.url).searchParams.get('key');
a.kill(); await sleep(400);
const b = spawn('node', [SERVER], { env: { ...env, BRAINSTORM_DIR: path.join(dir, 's2') } });
let outB = ''; b.stdout.on('data', d => outB += d.toString());
for (let i = 0; i < 60 && !outB.includes('server-started'); i++) await sleep(50);
const infoB = firstServerStarted(outB);
const ws = new WebSocket(`ws://localhost:${infoB.port}/?key=${keyA}`, {
headers: { Origin: `http://localhost:${infoB.port}` }
});
const opened = await new Promise(resolve => {
ws.on('open', () => resolve(true));
ws.on('error', () => resolve(false));
setTimeout(() => resolve(false), 1500);
});
try {
assert.strictEqual(infoB.port, infoA.port, 'restart should reuse same port');
assert(opened, 'stored key should authenticate WS after restart');
} finally {
try { ws.close(); } catch (e) {}
b.kill(); await sleep(100);
fs.rmSync(dir, { recursive: true, force: true });
}
});

This test may already pass once Tasks 2 and 3 are implemented. 如果 it passes before code changes, keep it as coverage but do not call it RED. The real browser reconnect behavior is primarily covered by Task 3 plus final manual/headless browser verification.

  • 步骤 2: 验证 behavior

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node lifecycle.test.js

预期 after Tasks 2 and 3: lifecycle tests pass. 如果 this fails, fix the auth/restart path before continuing.

文件:

  • 修改: tests/brainstorm-server/lifecycle.test.js

  • 修改: skills/brainstorming/scripts/start-server.sh

  • 修改: skills/brainstorming/scripts/stop-server.sh

  • 步骤 1: Reproduce shell lint 失败

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers
scripts/lint-shell.sh skills/brainstorming/scripts/start-server.sh skills/brainstorming/scripts/stop-server.sh tests/brainstorm-server/stop-server.test.sh

预期 当前 失败:

SC2164: skills/brainstorming/scripts/start-server.sh line 128: cd "$SCRIPT_DIR"
SC2034: skills/brainstorming/scripts/start-server.sh line 166: for i in {1..50}
SC2034: skills/brainstorming/scripts/stop-server.sh line 57: for i in {1..20}
  • 步骤 2: Fix shell lint minimally

In skills/brainstorming/scripts/start-server.sh, change:

Terminal window
cd "$SCRIPT_DIR"

到:

Terminal window
cd "$SCRIPT_DIR" || exit 1

Change unused loop variables from i to _ where they are not read:

Terminal window
for _ in {1..50}; do

In skills/brainstorming/scripts/stop-server.sh, change:

Terminal window
for i in {1..20}; do

到:

Terminal window
for _ in {1..20}; do
  • 步骤 3: Fix lifecycle start-server hang

In tests/brainstorm-server/lifecycle.test.js, update the start-server.sh --idle-timeout-minutes sets the timeout test command:

const out = execFileSync('bash', [START, '--project-dir', dir, '--idle-timeout-minutes', '5', '--background'], { encoding: 'utf8' });

This keeps the test from hanging when CODEX_CI triggers start-server foreground mode.

  • 步骤 4: 验证 lint and lifecycle

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers
scripts/lint-shell.sh skills/brainstorming/scripts/start-server.sh skills/brainstorming/scripts/stop-server.sh tests/brainstorm-server/stop-server.test.sh
cd tests/brainstorm-server
node lifecycle.test.js

预期: shell lint exits 0; lifecycle tests exit 0 without hanging.

文件:

  • 修改: .gitignore

  • 步骤 1: 验证 当前 ignore gap

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers
git check-ignore .superpowers/brainstorm/.last-token || true

预期 当前 output: no matching ignore rule.

  • 步骤 2: 添加 ignore rule

添加 this line to .gitignore:

.superpowers/
  • 步骤 3: 验证 GREEN

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers
git check-ignore .superpowers/brainstorm/.last-token

预期 output:

.superpowers/brainstorm/.last-token

文件:

  • No code changes in this task.

  • 步骤 1: 运行 focused suites

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
node auth.test.js
node helper.test.js
node server.test.js
node lifecycle.test.js

预期: all four commands exit 0.

  • 步骤 2: 运行 full brainstorm-server suite

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
npm test

预期: all tests pass, including ws-protocol, helper, auth, 服务器, lifecycle, and stop-server.

  • 步骤 3: Repeat suite for lifecycle/watch flake

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers/tests/brainstorm-server
for i in 1 2 3; do npm test || exit 1; done

预期: all three repeats pass without hanging.

  • 步骤 4: 运行 shell lint

运行:

Terminal window
cd /Users/drewritter/prime-rad/superpowers
scripts/lint-shell.sh skills/brainstorming/scripts/start-server.sh skills/brainstorming/scripts/stop-server.sh tests/brainstorm-server/stop-server.test.sh

预期: exits 0.

文件:

  • No code changes in this task.

  • 步骤 1: Recreate the cross-origin attacker probe

使用 the previous scratch probe if 可用:

Terminal window
node /tmp/superpowers-pr1720-security-drewritter/probe-pr1720.cjs

如果 the scratch probe is 不可用, recreate a minimal probe under /tmp that:

  • starts the companion with a fixed token
  • 加载 the keyed URL in headless Chrome
  • starts an attacker page on a different localhost port
  • attempts new WebSocket('ws://localhost:<companion-port>/')
  • sends {"type":"choice","choice":"attacker-injected"}
  • checks state/events

预期 after fixes:

  • keyless and wrong-key HTTP still return 403

  • same-origin helper reaches Connected

  • cross-origin WebSocket does not open

  • state/events does not contain attacker-injected

  • symlink-to-server-info returns 404

  • keyed browser load ends on bare /

  • 步骤 2: Re-run manual/browser flow only after automated probes pass

Manual flow:

  1. 启动 the companion with --project-dir --open
  2. push a screen
  3. confirm URL strips to /
  4. confirm status reaches Connected
  5. click a choice and verify state/events
  6. 停止 and 重启 same 项目
  7. verify the open tab reconnects 自动

预期: all steps pass without 手动 URL reload.

  • Spec coverage: every 设计 需求 maps to at least one task.
  • Placeholder scan: this 计划 contains no unresolved placeholder markers or unspecified edge-case steps.
  • TDD order: every production change 任务 starts with a focused failing test or a command that demonstrates the 当前 failure.
  • Trust model: the 计划 preserves trusted same-origin screen JavaScript and future same-origin vendored libraries.
  • No-commit rule: execution does not commit unless Drew explicitly asks.
-
0:000:00