mirror of
https://github.com/garrytan/gstack.git
synced 2026-06-17 23:30:09 +02:00
5373bc32ae
Extends bin/gstack-config to support the brain-aware planning layer:
KEY VALIDATION (T5):
Plain alphanumeric/underscore now extended to allow @<hex-hash> suffix.
Required for per-endpoint namespaced keys (brain_trust_policy@<sha8>,
user_slug_at_<sha8>). Keys without the suffix still validate as before.
VALUE WHITELISTING (D4 / D11):
brain_trust_policy@* values gated to personal | shared | unset.
Unknown values warn + default to unset (defense against typos).
NEW DEFAULTS (lookup_default):
brain_trust_policy@* -> unset
salience_allowlist -> '' (resolver uses SALIENCE_DEFAULT_ALLOWLIST)
user_slug_at_* -> '' (resolve-user-slug fills + persists on demand)
NEW SUBCOMMANDS:
endpoint-hash — print sha8 of active gbrain MCP URL from
~/.claude.json. Collision check escalates to sha16
when a prior endpoint stored at the same sha8
would conflict (T10 defensive default).
resolve-user-slug — walks D4 A3 identity chain:
1. mcp__gbrain__whoami.client_name
2. $USER env var
3. sha8(git config user.email)
4. anonymous-<sha8(hostname)>
Persists result on first call so subsequent
calls are stable across sessions.
test/user-slug-fallback.test.ts: 14 tests covering endpoint-hash output
shape, fallback chain ordering, persistence, brain_trust_policy
namespace value validation + per-endpoint isolation, and key validator
extension for @-suffixed keys.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
162 lines
6.4 KiB
TypeScript
162 lines
6.4 KiB
TypeScript
/**
|
|
* User-slug identity resolution chain (T16 / D4 A3).
|
|
*
|
|
* Verifies the gstack-config resolve-user-slug subcommand walks the
|
|
* documented fallback chain:
|
|
* 1. mcp__gbrain__whoami.client_name (skipped when gbrain not on PATH)
|
|
* 2. $USER env var
|
|
* 3. sha8($(git config user.email))
|
|
* 4. anonymous-<sha8(hostname)>
|
|
*
|
|
* Result is persisted under user_slug_at_<endpoint-hash> for stability.
|
|
* Test isolation via GSTACK_HOME and HOME env overrides.
|
|
*
|
|
* Gate-tier, free, ~50ms.
|
|
*/
|
|
|
|
import { describe, test, expect, beforeEach, afterEach } from 'bun:test';
|
|
import { mkdtempSync, existsSync, readFileSync, writeFileSync, rmSync, mkdirSync } from 'fs';
|
|
import { join } from 'path';
|
|
import { tmpdir } from 'os';
|
|
import { spawnSync } from 'child_process';
|
|
|
|
const REPO_ROOT = process.cwd();
|
|
const CONFIG_BIN = join(REPO_ROOT, 'bin', 'gstack-config');
|
|
|
|
let TMP_HOME: string;
|
|
const ORIGINAL = {
|
|
HOME: process.env.HOME,
|
|
GSTACK_HOME: process.env.GSTACK_HOME,
|
|
USER: process.env.USER,
|
|
};
|
|
|
|
function runConfig(args: string[], extraEnv: Record<string, string> = {}): { stdout: string; status: number; stderr: string } {
|
|
const result = spawnSync(CONFIG_BIN, args, {
|
|
encoding: 'utf-8',
|
|
env: {
|
|
...process.env,
|
|
...extraEnv,
|
|
},
|
|
timeout: 5000,
|
|
});
|
|
return { stdout: result.stdout || '', status: result.status ?? -1, stderr: result.stderr || '' };
|
|
}
|
|
|
|
beforeEach(() => {
|
|
TMP_HOME = mkdtempSync(join(tmpdir(), 'gstack-user-slug-test-'));
|
|
process.env.GSTACK_HOME = TMP_HOME;
|
|
});
|
|
|
|
afterEach(() => {
|
|
for (const [k, v] of Object.entries(ORIGINAL)) {
|
|
if (v !== undefined) process.env[k] = v;
|
|
else delete (process.env as Record<string, unknown>)[k];
|
|
}
|
|
try { rmSync(TMP_HOME, { recursive: true, force: true }); } catch { /* best effort */ }
|
|
});
|
|
|
|
describe('endpoint-hash subcommand', () => {
|
|
test('returns deterministic 8-char hex or literal "local"', () => {
|
|
const result = runConfig(['endpoint-hash'], { GSTACK_HOME: TMP_HOME });
|
|
expect(result.status).toBe(0);
|
|
const out = result.stdout.trim();
|
|
expect(out === 'local' || /^[a-f0-9]{8}$/.test(out) || /^[a-f0-9]{16}$/.test(out)).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe('resolve-user-slug fallback chain', () => {
|
|
test('uses $USER when set (layer 2)', () => {
|
|
const result = runConfig(['resolve-user-slug'], { GSTACK_HOME: TMP_HOME, USER: 'alice-test' });
|
|
expect(result.status).toBe(0);
|
|
expect(result.stdout.trim()).toBe('alice-test');
|
|
});
|
|
|
|
test('lowercases + dash-normalizes $USER', () => {
|
|
const result = runConfig(['resolve-user-slug'], { GSTACK_HOME: TMP_HOME, USER: 'Alice Test' });
|
|
expect(result.status).toBe(0);
|
|
// Spaces become dashes, uppercase becomes lowercase
|
|
expect(result.stdout.trim()).toMatch(/^alice-test$/i);
|
|
});
|
|
|
|
test('falls through past empty $USER to git email or anonymous', () => {
|
|
const result = runConfig(['resolve-user-slug'], { GSTACK_HOME: TMP_HOME, USER: '' });
|
|
expect(result.status).toBe(0);
|
|
const slug = result.stdout.trim();
|
|
expect(slug.length).toBeGreaterThan(0);
|
|
// Should be either email-<sha8> or anonymous-<sha8>
|
|
expect(slug).toMatch(/^(email-|anonymous-)[a-f0-9]+$|^[a-zA-Z0-9-]+$/);
|
|
});
|
|
|
|
test('persists resolution to user_slug_at_<hash> on first call', () => {
|
|
runConfig(['resolve-user-slug'], { GSTACK_HOME: TMP_HOME, USER: 'persisttest' });
|
|
const configFile = join(TMP_HOME, 'config.yaml');
|
|
expect(existsSync(configFile)).toBe(true);
|
|
const content = readFileSync(configFile, 'utf-8');
|
|
expect(content).toMatch(/^user_slug_at_[a-f0-9]+:\s+persisttest/m);
|
|
});
|
|
|
|
test('subsequent calls return same slug (stable across sessions)', () => {
|
|
const first = runConfig(['resolve-user-slug'], { GSTACK_HOME: TMP_HOME, USER: 'stabletest' });
|
|
const second = runConfig(['resolve-user-slug'], { GSTACK_HOME: TMP_HOME, USER: 'changed-after' });
|
|
// Second call ignores new $USER because the slug was already persisted.
|
|
expect(first.stdout.trim()).toBe('stabletest');
|
|
expect(second.stdout.trim()).toBe('stabletest');
|
|
});
|
|
});
|
|
|
|
describe('brain_trust_policy@<hash> namespace', () => {
|
|
test('default value is "unset"', () => {
|
|
const result = runConfig(['get', 'brain_trust_policy@deadbeef'], { GSTACK_HOME: TMP_HOME });
|
|
expect(result.status).toBe(0);
|
|
expect(result.stdout).toBe('unset');
|
|
});
|
|
|
|
test('set + get roundtrip works', () => {
|
|
const setResult = runConfig(['set', 'brain_trust_policy@deadbeef', 'personal'], { GSTACK_HOME: TMP_HOME });
|
|
expect(setResult.status).toBe(0);
|
|
const getResult = runConfig(['get', 'brain_trust_policy@deadbeef'], { GSTACK_HOME: TMP_HOME });
|
|
expect(getResult.stdout).toBe('personal');
|
|
});
|
|
|
|
test('invalid value falls back to unset with warning', () => {
|
|
const result = runConfig(['set', 'brain_trust_policy@deadbeef', 'invalid-value'], { GSTACK_HOME: TMP_HOME });
|
|
expect(result.status).toBe(0);
|
|
expect(result.stderr).toContain('not recognized');
|
|
const getResult = runConfig(['get', 'brain_trust_policy@deadbeef'], { GSTACK_HOME: TMP_HOME });
|
|
expect(getResult.stdout).toBe('unset');
|
|
});
|
|
|
|
test('shared value accepted', () => {
|
|
runConfig(['set', 'brain_trust_policy@deadbeef', 'shared'], { GSTACK_HOME: TMP_HOME });
|
|
const getResult = runConfig(['get', 'brain_trust_policy@deadbeef'], { GSTACK_HOME: TMP_HOME });
|
|
expect(getResult.stdout).toBe('shared');
|
|
});
|
|
|
|
test('per-endpoint policies dont collide', () => {
|
|
runConfig(['set', 'brain_trust_policy@aaaaaaaa', 'personal'], { GSTACK_HOME: TMP_HOME });
|
|
runConfig(['set', 'brain_trust_policy@bbbbbbbb', 'shared'], { GSTACK_HOME: TMP_HOME });
|
|
const a = runConfig(['get', 'brain_trust_policy@aaaaaaaa'], { GSTACK_HOME: TMP_HOME });
|
|
const b = runConfig(['get', 'brain_trust_policy@bbbbbbbb'], { GSTACK_HOME: TMP_HOME });
|
|
expect(a.stdout).toBe('personal');
|
|
expect(b.stdout).toBe('shared');
|
|
});
|
|
});
|
|
|
|
describe('key validation', () => {
|
|
test('rejects keys with disallowed characters', () => {
|
|
const result = runConfig(['get', 'bad-key'], { GSTACK_HOME: TMP_HOME });
|
|
expect(result.status).not.toBe(0);
|
|
expect(result.stderr).toContain('alphanumeric');
|
|
});
|
|
|
|
test('accepts plain alphanumeric/underscore keys', () => {
|
|
const result = runConfig(['get', 'proactive'], { GSTACK_HOME: TMP_HOME });
|
|
expect(result.status).toBe(0);
|
|
});
|
|
|
|
test('accepts @<hex-hash> suffix on key', () => {
|
|
const result = runConfig(['get', 'brain_trust_policy@abc123ff'], { GSTACK_HOME: TMP_HOME });
|
|
expect(result.status).toBe(0);
|
|
});
|
|
});
|