README changes: - Rewrote the "Cross-machine memory with GBrain sync" section into "GBrain — persistent knowledge for your coding agent." Covers the three /setup-gbrain paths (Supabase existing URL, auto-provision, PGLite local), MCP registration, per-remote trust triad, and the (still-separate) memory sync feature. - Added /setup-gbrain row to the skills table pointing at the full guide. - Added /setup-gbrain to both skill-list install snippets. - Added USING_GBRAIN_WITH_GSTACK.md to the Docs table. New doc (USING_GBRAIN_WITH_GSTACK.md): - All three setup paths with trust-surface caveats - MCP registration details (and honest Claude-Code-v1 scoping) - Per-remote trust triad semantics + how to change a policy - Switching engines (PGLite ↔ Supabase) via --switch - GStack memory sync + its relationship to the gbrain knowledge base - /setup-gbrain --cleanup-orphans for orphan Supabase projects - Full command + flag reference, every bin helper, every env var - Security model: what's enforced in code, what's enforced by the leak harness, and the honest limits of v1 - Troubleshooting: PATH shadowing, direct-connection URL reject, auto-provision timeout, stale lock, policy file hand-edits, migrate hang - Why-this-design section explaining the non-obvious choices Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
19 KiB
Using GBrain with GStack
Your coding agent, with a memory it actually keeps.
GBrain is a persistent knowledge base designed for AI agents. It stores what your agent learns, what you've decided, what worked and what didn't, and lets the agent search all of it on demand. GStack gives you a one-command path from zero to "gbrain is running, and my agent can call it" — with paths for try-it-local, share-with-your-team, and everything between.
This is the full monty: every scenario, every flag, every helper bin, every troubleshooting step. For the quick pitch, see the README's GBrain section. For error codes and sync-specific issues, see docs/gbrain-sync.md.
The one-command install
/setup-gbrain
That's it. The skill detects your current state, asks three questions at most, and walks you through install, init, MCP registration for Claude Code, and per-repo trust policy. On a clean Mac with nothing installed it finishes in under five minutes. On a Mac where something's already set up it takes seconds (it detects the existing state and skips done work).
The three paths
You pick one when the skill asks "Where should your brain live?"
Path 1: Supabase, you already have a connection string
Best for: you (or a teammate's cloud agent) already provisioned a Supabase brain and you want this local machine to use the same data.
What happens: Paste the Session Pooler URL (Settings → Database → Connection Pooler → Session → copy URI, port 6543). The skill reads it with echo off, shows you a redacted preview (aws-0-us-east-1.pooler.supabase.com:6543/postgres — host visible, password masked), hands it to gbrain init via the GBRAIN_DATABASE_URL environment variable, and the URL is never written to argv or your shell history.
Trust warning: Pasting this URL gives your local Claude Code full read/write access to every page in the shared brain. If that's not the trust level you want, pick PGLite local (Path 3) instead and accept the brains are disjoint.
Path 2a: Supabase, auto-provision a new project
Best for: fresh Supabase account, you want a clean new project with zero clicking.
What happens: You paste a Supabase Personal Access Token (PAT). The skill shows you the scope disclosure first — the token grants full access to every project in your Supabase account, not just the one we're about to create. It lists your organizations, asks which one and which region (default us-east-1), generates a database password, calls POST /v1/projects, polls GET /v1/projects/{ref} every 5 seconds until the project is ACTIVE_HEALTHY (180s timeout), fetches the pooler URL, hands it to gbrain init. End-to-end: ~90 seconds.
At the end: explicit reminder to revoke the PAT at https://supabase.com/dashboard/account/tokens. The skill already discarded it from memory.
If you Ctrl-C mid-provision: The SIGINT trap prints your in-flight project ref + a resume command. You can delete the orphan at the Supabase dashboard, or run /setup-gbrain --resume-provision <ref> to pick up where you left off.
Path 2b: Supabase, create manually
Best for: you'd rather click through supabase.com yourself than paste a PAT.
What happens: The skill walks you through the four manual steps (signup → new project → wait ~2 min → copy Session Pooler URL), then takes over from Path 1's paste step. Same security treatment as Path 1.
Path 3: PGLite local
Best for: try-it-first, no account, no cloud, no sharing. Or a dedicated "this Mac's brain" that stays isolated from any cloud agent.
What happens: gbrain init --pglite. Brain lives at ~/.gbrain/brain.pglite. No network calls. Done in 30 seconds.
This is the best first choice if you just want to see what gbrain feels like before committing to cloud. You can always migrate later with /setup-gbrain --switch.
MCP registration for Claude Code
By default the skill asks "Give Claude Code a typed tool surface for gbrain?" If you say yes, it runs:
claude mcp add gbrain -- gbrain serve
That registers gbrain's stdio MCP server with Claude Code. Now gbrain search, gbrain put_page, gbrain get_page, etc. show up as first-class tools in every session, not bash shell-outs.
If claude is not on PATH, the skill skips MCP registration gracefully with a manual-register hint. The CLI resolver still works from any skill that shells out to gbrain — MCP is an upgrade, not a prerequisite.
Other local agents (Cursor, Codex CLI, etc.) need their own MCP registration. The skill is Claude-Code-targeted for v1; other hosts can register gbrain serve manually in their own MCP config.
Per-remote trust policy (the triad)
Every repo on your machine gets a policy decision: read-write, read-only, or deny.
- read-write — your agent can
gbrain searchfrom this repo's context AND write new pages back to the brain. Default for your own projects. - read-only — your agent can search the brain but never writes new pages from this repo's sessions. Ideal for multi-client consultants: search the shared brain, don't contaminate it with Client A's code while you're in Client B's repo.
- deny — no gbrain interaction at all. The repo is invisible to gbrain tooling.
The skill asks once per repo the first time you run a gstack skill there. After that the decision is sticky — every worktree + branch of the same git remote shares the same policy, so you set it once and it follows you.
SSH and HTTPS remote variants collapse to the same key: https://github.com/foo/bar.git and git@github.com:foo/bar.git are the same repo.
To change a policy:
/setup-gbrain --repo # re-prompt for this repo only
# Or directly:
~/.claude/skills/gstack/bin/gstack-gbrain-repo-policy set "github.com/foo/bar" read-only
To see every policy:
~/.claude/skills/gstack/bin/gstack-gbrain-repo-policy list
Storage: ~/.gstack/gbrain-repo-policy.json, mode 0600, schema-versioned so future migrations stay deterministic.
Switching engines later
Picked PGLite and now want to join a team brain? One command:
/setup-gbrain --switch
The skill runs gbrain migrate --to supabase --url "$URL" wrapped in timeout 180s. Migration is bidirectional (Supabase → PGLite also works) and lossless — pages, chunks, embeddings, links, tags, and timeline all copy. Your original brain is preserved as a backup.
If migration hangs: another gstack session may be holding a lock on the source brain. The timeout fires at 3 minutes with an actionable message. Close other workspaces and re-run.
GStack memory sync (a separate concern)
This is different from gbrain itself. Your gstack state (~/.gstack/ — learnings, plans, retros, timeline, developer profile) is machine-local by default. "GStack memory sync" optionally pushes a curated, secret-scanned subset to a private git repo so your memory follows you across machines — and, if you're running gbrain, that git repo becomes indexable there too.
Turn it on with:
gstack-brain-init
You'll get a one-time privacy prompt: everything allowlisted / artifacts only (plans, designs, retros, learnings — skip behavioral data like timelines) / off. Every skill run syncs the queue at start and end — no daemon, no background process.
Secret-shaped content (AWS keys, GitHub tokens, PEM blocks, JWTs, bearer tokens) is blocked from sync before it leaves your machine.
On a new machine: Copy ~/.gstack-brain-remote.txt over, run gstack-brain-restore, and yesterday's learnings surface on today's laptop.
Full guide: docs/gbrain-sync.md. Error index: docs/gbrain-sync-errors.md.
/setup-gbrain offers to wire this up for you at the end of initial setup — it's one more AskUserQuestion, and it integrates with the same private-repo infrastructure.
Cleanup orphan projects
If you Ctrl-C'd mid-provision, tried three different names before settling on one, or otherwise accumulated gbrain-shaped Supabase projects you don't use, there's a subcommand for that:
/setup-gbrain --cleanup-orphans
The skill re-collects a PAT (one-time, discarded after), lists every project in your Supabase account whose name starts with gbrain and whose ref doesn't match your active ~/.gbrain/config.json pooler URL. For each orphan it asks per-project: "Delete orphan project <ref> (<name>, created <date>)?" — no batching, no "delete all" shortcut. The active brain is never offered for deletion.
Command + flag reference
/setup-gbrain entry modes
| Invocation | What it does |
|---|---|
/setup-gbrain |
Full flow: detect state, pick path, install, init, MCP, policy, optional memory-sync |
/setup-gbrain --repo |
Flip the per-remote trust policy for the current repo only |
/setup-gbrain --switch |
Migrate engine (PGLite ↔ Supabase) without re-running the other steps |
/setup-gbrain --resume-provision <ref> |
Resume a path-2a auto-provision that was interrupted during polling |
/setup-gbrain --cleanup-orphans |
List + per-project delete of orphan Supabase projects |
Bin helpers (for scripting)
| Bin | Purpose |
|---|---|
gstack-gbrain-detect |
Emit current state as JSON: gbrain on PATH, version, config engine, doctor status, sync mode |
gstack-gbrain-install |
Detect-first installer (probes ~/git/gbrain, ~/gbrain, then fresh clone). Has --dry-run and --validate-only flags. PATH-shadow check exits 3 with remediation menu. |
gstack-gbrain-lib.sh |
Sourced, not executed. Provides read_secret_to_env VARNAME "prompt" [--echo-redacted "<sed-expr>"] |
gstack-gbrain-supabase-verify |
Structural URL check. Rejects direct-connection URLs (db.*.supabase.co:5432) with exit 3 |
gstack-gbrain-supabase-provision |
Management API wrapper. Subcommands: list-orgs, create, wait, pooler-url, list-orphans, delete-project. All require SUPABASE_ACCESS_TOKEN in env. create and pooler-url also require DB_PASS. --json mode available on every subcommand. |
gstack-gbrain-repo-policy |
Per-remote trust triad. Subcommands: get, set, list, normalize |
gbrain CLI (upstream tool)
Gbrain itself ships with these that gstack wraps:
| Command | Purpose |
|---|---|
gbrain init --pglite |
Initialize a local PGLite brain |
gbrain init --non-interactive |
Initialize via env (GBRAIN_DATABASE_URL or DATABASE_URL). Never pass a URL as argv — it'll leak to shell history. |
gbrain doctor --json |
Health check. Returns `{status: "ok" |
gbrain migrate --to supabase --url ... |
Move a PGLite brain to Supabase (lossless, preserves source as backup) |
gbrain migrate --to pglite |
Reverse migration |
gbrain search "query" |
Search the brain |
gbrain put_page --title "..." --tags "a,b" <<<"content" |
Write a page |
gbrain get_page "<slug>" |
Fetch a page |
gbrain serve |
Start the MCP stdio server (used by claude mcp add) |
Config files + state
| Path | What lives there |
|---|---|
~/.gbrain/config.json |
Engine (pglite/postgres), database URL or path, API keys. Mode 0600. Written by gbrain init. |
~/.gstack/gbrain-repo-policy.json |
Per-remote trust triad. Schema v2. Mode 0600. |
~/.gstack/.setup-gbrain.lock.d |
Concurrent-run lock (atomic mkdir). Released on normal exit + SIGINT. |
~/.gstack/.brain-queue.jsonl |
Pending sync entries for gstack memory sync |
~/.gstack/.brain-last-push |
Timestamp of last sync push (for /health scoring) |
~/.gstack-brain-remote.txt |
URL of your gstack memory sync remote (safe to copy between machines) |
~/.gstack/.setup-gbrain-inflight.json |
Reserved for future --resume-provision persisted state |
Environment variables
| Var | Where it's read | What it does |
|---|---|---|
SUPABASE_ACCESS_TOKEN |
gstack-gbrain-supabase-provision |
PAT for Management API calls. Discarded after each setup run. |
DB_PASS |
gstack-gbrain-supabase-provision (create, pooler-url) |
Generated DB password. Never in argv. |
GBRAIN_DATABASE_URL |
gbrain init, gbrain doctor, etc. |
Postgres connection string (Supabase pooler URL for us). Env takes precedence over ~/.gbrain/config.json. |
DATABASE_URL |
gbrain init (fallback) |
Same semantics as GBRAIN_DATABASE_URL; checked second. |
SUPABASE_API_BASE |
gstack-gbrain-supabase-provision |
Override the Management API host. Used by tests to point at a mock server. |
GBRAIN_INSTALL_DIR |
gstack-gbrain-install |
Override default install path (~/gbrain) |
GSTACK_HOME |
every bin helper | Override ~/.gstack state dir. Heavy test use. |
Security model
One rule for every secret this skill touches: env var only, never argv, never logged, never written to disk by us. The only persistent storage is gbrain's own ~/.gbrain/config.json at mode 0600, which is gbrain's discipline, not ours.
Enforced in code:
- CI grep test in
test/skill-validation.test.tsfails the build if$SUPABASE_ACCESS_TOKENor$GBRAIN_DATABASE_URLappears in an argv position - CI grep test fails if
--insecure,-k, orNODE_TLS_REJECT_UNAUTHORIZED=0appear inbin/gstack-gbrain-supabase-provision set +xat the top of the provision helper prevents debug tracing from leaking PAT- Telemetry payload contains only enumerated categorical values (scenario, install result, MCP opt-in, trust tier) — never free-form strings that could contain secrets
Enforced via tests:
test/secret-sink-harness.test.tsruns every secret-handling bin with a seeded secret and asserts the seed never appears in any captured channel (stdout, stderr, files under$HOME, telemetry JSONL). Four match rules per seed: exact, URL-decoded, first-12-char prefix, base64.- Positive controls in the same test file deliberately leak seeds in every covered channel and assert the harness catches each one. Without the positive controls, a harness that silently under-reports would look identical to a working harness.
What you can still leak (the honest limits of v1):
- If you paste a secret into a normal chat message outside
read -s, it's in the conversation transcript and any host-side logging - The leak harness doesn't dump subprocess environment — a bin that
env >> ~/.logwould evade detection (no bin in v1 does this; grep tests prevent it) - Your shell's own
HISTFILEbehavior is your shell's, not ours — we never pass secrets to argv so they don't land there via our code, but nothing stops you from pasting one into a rawcurlcommand yourself
Troubleshooting
"PATH SHADOWING DETECTED" during install
Another gbrain binary is earlier in PATH than the one the installer just linked. The installer's version check caught it. Fix one of:
rm $(which gbrain)if you don't need the other one- Prepend
~/.bun/binto PATH in your shell rc so the linked binary wins - Set
GBRAIN_INSTALL_DIRto the shadowing binary's install directory and re-run
Then re-run /setup-gbrain.
"rejected direct-connection URL"
You pasted a db.<ref>.supabase.co:5432 URL. Those are IPv6-only and fail in most environments. Use the Session Pooler URL instead: Supabase dashboard → Settings → Database → Connection Pooler → Session → copy URI (port 6543).
Auto-provision times out at 180s
The Supabase project is still initializing. Your ref was printed in the exit message. Wait a minute, then:
/setup-gbrain --resume-provision <ref>
The skill re-collects a PAT, skips project creation, resumes polling.
"Another /setup-gbrain instance is running"
You have a stale lock directory. If you're sure no other instance is actually running:
rm -rf ~/.gstack/.setup-gbrain.lock.d
Then re-run.
"No cross-model tension" on policy file
You edited ~/.gstack/gbrain-repo-policy.json by hand with legacy allow values? No problem. On the next read, gstack auto-migrates allow → read-write and adds _schema_version: 2. One log line on stderr, idempotent, deterministic.
gbrain doctor says "warnings"
/health treats that as yellow, not red. Check gbrain doctor --json | jq .checks to see which sub-checks are warning. Typical causes: resolver MECE overlap (skill names clashing) or DB connection not yet configured.
Switching PGLite → Supabase hangs
Another gstack session in a sibling Conductor workspace may be holding a lock on your local PGLite file via its preamble's gstack-brain-sync call. Close other workspaces, re-run /setup-gbrain --switch. The timeout is bounded at 180s so you'll never actually wait forever.
Why this design
Why per-remote trust triad and not binary allow/deny? Multi-client consultants need search without write-back. A freelance dev working on Client A in the morning and Client B in the afternoon can't let A's code insights leak into a brain Client B can search. Read-only solves that cleanly.
Why not bundle gbrain into gstack? Gbrain is a separate, actively-developed project with its own release cadence, schema migrations, and MCP surface. Bundling would mean gstack has to gate gbrain updates, which slows gbrain improvements from reaching users. Separate-but-integrated lets each ship on its own cadence.
Why gbrain init --non-interactive via env var and not a flag? Connection strings contain database passwords. Passing them as argv lands the password in ps, shell history, and process listings. Env-var handoff keeps the secret in process memory only. Gbrain supports both GBRAIN_DATABASE_URL and DATABASE_URL; we use the former to avoid collisions with non-gbrain tooling.
Why fail-hard on PATH shadowing instead of warn-and-continue? A shadowed gbrain means every subsequent command calls a different binary than the one we just installed. That's a silent version-drift bug that surfaces as mysterious feature gaps weeks later. Setup skills have one job — set up a working environment. Refusing to install into a broken one is the setup-skill-correct behavior.
Why not auto-import every repo? Privacy + noise. An auto-import preamble hook that ingests every repo you touch would: (a) leak work code into a shared brain without consent, and (b) clog search with throwaway repos. The per-remote policy makes ingestion an explicit, per-repo decision. /setup-gbrain doesn't install any auto-import hook today — but the policy store is forward-compatible for one later.
Related skills + next steps
/health— includes a GBrain dimension (doctor status, sync queue depth, last-push age) in its 0-10 composite score. The dimension is omitted when gbrain isn't installed; running/healthon a non-gbrain machine doesn't penalize that choice./gstack-upgrade— keeps gstack itself up to date. Does NOT upgrade gbrain independently. To bump gbrain, updatePINNED_COMMITinbin/gstack-gbrain-installand re-run/setup-gbrain./retro— weekly retrospective pulls learnings and plans from your gbrain when memory sync is on, letting the retro reference cross-machine history.
Run /setup-gbrain and see what sticks.