Files
gstack/BROWSER.md
T
Garry Tan e8893a18b1 v1.20.0.0 feat: browser-skills runtime + gbrain-support carryover (#1233)
* feat(gbrain-sync): queue primitives + writer shims

Adds bin/gstack-brain-enqueue (atomic append to sync queue) and
bin/gstack-jsonl-merge (git merge driver, ts-sort with SHA-256 fallback).
Wires one backgrounded enqueue call into learnings-log, timeline-log,
review-log, and developer-profile --migrate. question-log and
question-preferences stay local per Codex v2 decision.

gstack-config gains gbrain_sync_mode (off/artifacts-only/full) and
gbrain_sync_mode_prompted keys, plus GSTACK_HOME env alignment so
tests don't leak into real ~/.gstack/config.yaml.

* feat(gbrain-sync): --once drain + secret scan + push

bin/gstack-brain-sync is the core sync binary. Subcommands: --once
(drain queue, allowlist-filter, privacy-class-filter, secret-scan
staged diff, commit with template, push with fetch+merge retry),
--status, --skip-file <path>, --drop-queue --yes, --discover-new
(cursor-based detection of artifact writes that skip the shim).

Secret regex families: AWS keys, GitHub tokens (ghp_/gho_/ghu_/ghs_/
ghr_/github_pat_), OpenAI sk-, PEM blocks, JWTs, bearer-token-in-JSON.
On hit: unstage, preserve queue, print remediation hint (--skip-file
or edit), exit clean. No daemon — invoked by preamble at skill
boundaries.

* feat(gbrain-sync): init, restore, uninstall, consumer registry

bin/gstack-brain-init: idempotent first-run. git init ~/.gstack/,
.gitignore=*, canonical .brain-allowlist + .brain-privacy-map.json,
pre-commit secret-scan hook (defense-in-depth), merge driver registration
via git config, gh repo create --private OR arbitrary --remote <url>,
initial push, ~/.gstack-brain-remote.txt for new-machine discovery,
GBrain consumer registration via HTTP POST.

bin/gstack-brain-restore: safe new-machine bootstrap. Refuses clobber
of existing allowlisted files, clones to staging, rsync-copies tracked
files, re-registers merge drivers (required — not cloned from remote),
rehydrates consumers.json, prompts for per-consumer tokens.

bin/gstack-brain-uninstall: clean off-ramp. Removes .git + .brain-*
files + consumers.json + config keys. Preserves user data (learnings,
plans, retros, profile). Optional --delete-remote for GitHub repos.

bin/gstack-brain-consumer + bin/gstack-brain-reader (symlink alias):
registry management. Internal 'consumer' term; user-facing 'reader'
per DX review decision.

* feat(gbrain-sync): preamble block — privacy gate + boundary sync

scripts/resolvers/preamble/generate-brain-sync-block.ts emits bash that
runs at every skill invocation:
- Detects ~/.gstack-brain-remote.txt on machines without local .git
  and surfaces a restore-available hint (does NOT auto-run restore).
- Runs gstack-brain-sync --once at skill start to drain any pending
  writes (and at skill end via prose instruction).
- Once-per-day auto-pull (cached via .brain-last-pull) for append-only
  JSONL files.
- Emits BRAIN_SYNC: status line every skill run.

Also emits prose for the host LLM to fire the one-time privacy
stop-gate (full / artifacts-only / off) when gbrain is detected and
gbrain_sync_mode_prompted is false. Wired into preamble.ts composition.

* test(gbrain-sync): 27-test consolidated suite

test/brain-sync.test.ts covers:
- Config: validation, defaults, GSTACK_HOME env isolation
- Enqueue: no-op gates, skip list, concurrent atomicity, JSON escape
- JSONL merge driver: 3-way + ts-sort + SHA-256 fallback
- Init + sync: canonical file creation, merge driver registration,
  push-reject + fetch+merge retry path
- Init refuses different remote (idempotency)
- Cross-machine restore round-trip (machine A write → machine B sees)
- Secret scan across all 6 regex families (AWS, GH, OpenAI, PEM, JWT,
  bearer-JSON). --skip-file unblock remediation
- Uninstall removes sync config, preserves user data
- --discover-new idempotence via mtime+size cursor

Behaviors verified via integration smokes during implementation. Known
follow-up: bun-test 5s default timeout needs 30s wrapper for
spawnSync-heavy tests.

* docs(gbrain-sync): user guide + error lookup + README section

docs/gbrain-sync.md: setup walkthrough, privacy modes, cross-machine
workflow, secret protection, two-machine conflict handling, uninstall,
troubleshooting reference.

docs/gbrain-sync-errors.md: problem/cause/fix index for every
user-visible error. Patterned on Rust's error docs + Stripe's API
error reference.

README.md: 'Cross-machine memory with GBrain sync' section near the
top (discovery moment), plus docs-table entry.

* chore: bump version and changelog (v1.7.0.0)

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>

* chore: regenerate SKILL.md files for gbrain-sync preamble block

Re-runs bun run gen:skill-docs after adding generateBrainSyncBlock
to scripts/resolvers/preamble.ts in a2aa8a07. CI check-freshness
caught the drift. All 36 SKILL.md files regenerated with the new
skill-start bash block + privacy-gate prose + skill-end sync
instructions baked in.

* fix(test): session-awareness reads AskUserQuestion Format from a Tier 2+ SKILL.md

The test was reading ROOT/SKILL.md (browse skill, Tier 1) which never
contained '## AskUserQuestion Format' — that section is only emitted
for Tier 2+ skills by scripts/resolvers/preamble.ts. As a result the
agent was prompted with an empty format guide and only emitted
'RECOMMENDATION' intermittently, making the test flaky.

Pre-existing on main (same ROOT/SKILL.md shape there) — surfaced now
because the agent run didn't hit the RECOMMENDATION/recommend/option a
fallback strings in this particular attempt.

Fix: read from office-hours/SKILL.md (Tier 3, always has the section)
with a fallback that scans for the first top-level skill dir whose
SKILL.md contains the header. Future template moves won't break this
test again.

* feat(browse): domain-skills storage + state machine

New module browse/src/domain-skills.ts implements the per-site notes
the agent writes for itself, persisted as type:"domain" rows alongside
/learn's per-project learnings.

Three scopes layered: per-project default, global by explicit promotion.
Project-active shadows global for the same host.

State machine (T6 — codex outside-voice):
  quarantined --3 uses w/o flag--> active(project) --promote--> global
        ^                                |
        +----- classifier flag during use

- Append-only JSONL with O_APPEND for atomic small writes
- Tolerant parser drops partial trailing line on read
- Tombstone for deletes (compactor cleans up later)
- Version log per (host, scope) enables rollback
- Hostname derived from active tab top-level origin (T3 confused-deputy fix)
- writeSkill rejects classifier_score >= 0.85 with structured error

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(browse): domain-skills storage + state machine

14 tests covering:
- T3 hostname normalization (lowercase, www. strip, port/path/query strip,
  subdomain-exact preserved)
- T4 scope shadowing (per-project active shadows global for same host)
- T5 persistence (version monotonicity, tolerant parser drops partial line)
- T6 state machine (quarantined → active after N=3 uses, classifier-flag
  blocks promotion, save-time score >= 0.85 rejected)
- Rollback by version log (restore prior body, advance version counter)
- Tombstone deletion (read returns null after delete)

All 14 pass in 27ms via bun test.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(browse): $B domain-skill subcommands

Wire the domain-skills storage layer into the browse CLI as a META command:

  $B domain-skill save              save body from stdin or --from-file
                                     (host derived from active tab — T3)
  $B domain-skill list              list all skills visible to current project
  $B domain-skill show <host>       print skill body
  $B domain-skill edit <host>       open in $EDITOR
  $B domain-skill promote-to-global <host>  cross-project promotion (T4)
  $B domain-skill rollback <host> [--global]  restore prior version
  $B domain-skill rm <host> [--global]        tombstone

Save path runs L1-L3 content filters from content-security.ts (importable
in compiled binary, unlike L4 ML classifier — see CLAUDE.md). The L4
classifier scan happens in sidebar-agent at prompt-injection load time.

Output is structured (problem + cause + suggested-action) per DX D7.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(browse): $B cdp escape hatch — deny-default allowlist + two-tier mutex

Codex T2: flip CDP posture to deny-default. Allowed methods enumerated in
cdp-allowlist.ts with (scope: tab|browser, output: trusted|untrusted,
justification) per entry.

Initial allowlist (~25 methods) covers:
- Accessibility tree extraction (read-only)
- DOM/CSS inspection (read-only)
- Performance metrics
- Tracing
- Emulation viewport/UA override
- Page screenshot/PDF capture (output is binary, no marker injection vector)
- Network.enable/disable (no bodies/cookies — those are exfil surfaces)
- Runtime.getProperties (NO evaluate/callFunctionOn — those would be RCE)

Page.navigate is INTENTIONALLY NOT allowed; agents use $B goto which
goes through the URL blocklist.

Codex T7: two-tier mutex. tab-scoped methods take per-tab lock; browser-
scoped take global lock that blocks all tab locks. 5s acquire timeout
yields CDPMutexAcquireTimeout (no silent hangs). All lock acquires use
try/finally so errors don't leak the lock.

Path A from spike: uses Playwright's newCDPSession() per page. No second
WebSocket, no need for --remote-debugging-port. CDPSession is cached
per page in a WeakMap and cleared on page close.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(browse): CDP allowlist + two-tier mutex

13 tests:
- Allowlist linter: every entry has 4 required fields, no duplicates,
  justification length > 20 chars
- Deny-list verification: dangerous methods (Runtime.evaluate, Page.navigate,
  Network.getResponseBody, Browser.close, Target.attachToTarget, etc.) are
  NOT allowed (Codex T2 categories 4-7)
- Per-tab mutex serializes ops on same tab
- Per-tab mutex allows parallel ops across different tabs
- Global lock blocks tab locks; tab locks block global lock
- Acquire timeout yields CDPMutexAcquireTimeout (no silent hang)
- Timeout error names the tab id and the timeout budget

Also extends Network.disable justification to satisfy linter.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(browse): telemetry signals + project-slug helper

Lightweight telemetry per DX D9: piggybacks on ~/.gstack/analytics/ pattern.
Hostname + aggregate counters only, no body content. GSTACK_TELEMETRY_OFF=1
silences. Fire-and-forget — never blocks calling path.

Signals fired so far:
- domain_skill_saved {host, scope, state, bytes}
- domain_skill_save_blocked {host, reason}

(domain_skill_fired and cdp_method_* fired in subsequent commits.)

Also extracts project-slug resolution into project-slug.ts so server.ts
and domain-skill-commands.ts share one cached lookup.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(browse): sidebar prompt-context injection + CDP telemetry

server.ts spawnClaude now:
- Imports per-project domain skill matching the active tab's hostname
  via readDomainSkill()
- Wraps the body in UNTRUSTED EXTERNAL CONTENT envelope (so the L4
  classifier in sidebar-agent sees it at load time per Eng D4)
- Appends as <domain-skill source="..." host="..." version="..."> block
- Fires domain_skill_fired telemetry (host, source, version)
- Calls recordSkillUse fire-and-forget so the auto-promote-after-N=3
  state machine advances on each successful prompt injection

System prompt also gets a one-liner introducing $B domain-skill commands
to agents (DX D4 start-of-task discoverability hint).

cdp-bridge.ts fires:
- cdp_method_denied (drives next allow-list growth)
- cdp_method_lock_acquire_ms (P50/P99 quantile observability)
- cdp_method_called (allowed methods)

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(browse): telemetry module

3 tests covering:
- logTelemetry writes JSONL with ts injected
- GSTACK_TELEMETRY_OFF=1 silences all events
- logTelemetry never throws on disk failures

Uses GSTACK_HOME env var to redirect writes to a tmp dir; the telemetry
module reads HOME lazily so test mutations take effect.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* docs: domain-skills reference + error lookup table

docs/domain-skills.md mirrors the layered shape of docs/gbrain-sync.md
(DX D8): how agents use it, state machine, storage layout, security model
(L1-L3 + L4 layered defense), error reference table.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* docs(readme): browser-harness-js plug + domain-skills section

New "Domain skills + raw CDP escape hatch" section under "The sprint"
covering both v1.8.0.0 features. Plugs browser-use/browser-harness-js
as the no-rails alternative for users who want raw CDP without gstack's
security stack.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* chore: bump version and changelog (v1.8.0.0)

Branch-scoped bump on top of merged 1.7.0.0 base. CHANGELOG entry covers
the full v1.8.0.0 scope: $B domain-skill, $B cdp escape hatch, two-tier
mutex, telemetry signals, sidebar prompt-context injection. Includes
Codex outside-voice trail (7 of 20 findings resolved, 12 mooted by T1
scope drop).

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* todos: 7 follow-ups from v1.8.0.0 review trail

P1: Self-authoring $B commands with out-of-process worker isolation
    (Codex T1 deferred from v1.8.0.0 — needs real isolation design)
P2: Migrate /learn to SQLite (Codex T5 long-term primitive fix)
P2: Remove plan-mode handshake from /plan-devex-review (skill bug)
P3: GBrain skillpack publishing for domain-skills
P3: Replay/record demonstrated flows to domain-skills
P3: $B commands review batch-mode UX (alternative to inline approval)
P3: Heuristic command-gap watcher (DX D4 alternative C)

Each entry has the standard What/Why/Pros/Cons/Context/Effort/Priority/
Depends-on shape so anyone picking these up later has full context.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* fix(browse): lazy GSTACK_HOME resolution in domain-skills

Module-level constants (GLOBAL_FILE, derived path) were evaluated at
module-load and cached. When E2E and unit tests run in the same Bun
test pass and set GSTACK_HOME differently, the second test sees the
first test's path. Switch to lazy gstackHome() / globalFile() / projectFile()
helpers so process.env mutations take effect.

Mirrors the pattern already used in telemetry.ts.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(browse): E2E gate-tier tests for domain-skills + CDP

domain-skills-e2e.test.ts (4 tests):
- save derives host from active tab top-level origin (T3)
- save lands quarantined; list surfaces it
- readSkill returns null until 3 uses without flag promote to active (T6)
- save without an active page errors with structured guidance

cdp-e2e.test.ts (8 tests):
- Accessibility.getFullAXTree returns wrapped JSON (allowed, untrusted-output)
- Performance.getMetrics returns plain JSON (allowed, trusted-output)
- Runtime.evaluate DENIED with structured guidance (T2 RCE block)
- Page.navigate DENIED (must use $B goto for blocklist routing)
- Network.getResponseBody DENIED (exfil block)
- malformed JSON params surfaces clear error
- non Domain.method format surfaces clear error
- $B cdp help returns help text

Both files boot a real Chromium via BrowserManager.launch() and exercise
the dispatch handlers end-to-end. Total 12 E2E tests in <2s.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* docs: regenerate SKILL.md files with new $B commands

bun run gen:skill-docs picks up the domain-skill and cdp META_COMMANDS
entries added in commands.ts. Both top-level SKILL.md and browse/SKILL.md
now list the new commands in their Meta and Inspection tables.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(fixtures): regenerate ship SKILL.md golden baselines for v1.7.0.0

Pre-existing failures inherited from garrytan/gbrain-support: the GBrain
Sync preamble block (added in v1.7.0.0) appears in regenerated SKILL.md
output but the golden baselines in test/fixtures/golden/ were never
updated. Three failures fixed:

  golden-file regression > Claude ship skill matches golden baseline
  golden-file regression > Codex ship skill matches golden baseline
  golden-file regression > Factory ship skill matches golden baseline

Goldens regenerated by copying the current ship/SKILL.md, codex
.agents/skills/gstack-ship/SKILL.md, and .factory/skills/gstack-ship/SKILL.md
files. Diff is the v1.7.0.0 GBrain Sync preamble block + privacy stop-gate
(no behavioral changes — just preamble text).

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* fix(brain-sync): bearer-token regex catches values with leading space

Pre-existing bug from v1.7.0.0: the bearer-token-json secret pattern
required values matching [A-Za-z0-9_./+=-]{16,}, which rejected the
"Bearer <token>" form because the literal space after "Bearer" wasn't
in the character class. Real Authorization headers use "Bearer <token>"
syntax, and the test fixture
  '"authorization":"Bearer abcdef1234567890abcdef1234567890"'
sat unscanned despite being a leak-class secret.

One-character fix: add space to the value character class. Test
'gstack-brain-sync secret scan > blocks bearer-json' now passes.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(brain-sync): GSTACK_HOME isolation test compares mtime, not content

Pre-existing flaky test: the GSTACK_HOME-overrides-real-config test asserted
the real ~/.gstack/config.yaml does NOT contain "gbrain_sync_mode: full"
after the test. That fails for any user whose real config legitimately has
that key set from prior usage — the test's invariant is "the command did
not modify the real file," not "the real file lacks any specific value."

Switch to mtime + content snapshot: capture both BEFORE running the command,
then verify both are unchanged after. Also add a positive assertion that
the tmpHome config DID get the new key.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(skill-validation): exempt deliberate large fixtures from 2MB limit

Pre-existing failure: the "git tracks no files larger than 2MB" test
caught browse/test/fixtures/security-bench-haiku-responses.json (28.8MB
of replay data committed in v1.6.4.0 for security benchmark gate tests).

The test exists to catch accidentally-committed binaries (Mach-O dist
binaries, etc), not to forbid all large files. Add an explicit
LARGE_FIXTURE_EXEMPTIONS allowlist so deliberate replay fixtures pass
the gate while accidental binaries still fail.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(skill-token): mint scoped tokens per skill spawn

Wraps token-registry.createToken/revokeToken with skill-specific
clientId encoding (skill:<name>:<spawn-id>) and read+write defaults.
Skill scripts get a per-spawn capability token bound to browser-driving
commands; the daemon root token never leaves the harness.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(browse-client): SDK for browser-skill scripts

Thin wrapper over POST /command with bearer auth. Resolves daemon
port + token from GSTACK_PORT + GSTACK_SKILL_TOKEN env vars first
(set by $B skill run when spawning), falls back to .gstack/browse.json
for standalone debug runs.

Convenience methods cover the read+write surface skills typically need:
goto, click, fill, text, html, snapshot, links, forms, accessibility,
attrs, media, data, scroll, press, type, select, wait, hover, screenshot.
Low-level command(cmd, args) escape hatch for anything else.

This is the canonical SDK source. Each browser-skill ships a sibling
copy at <skill>/_lib/browse-client.ts so each skill is fully portable
and version-pinned.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(browser-skills): 3-tier storage helpers

listBrowserSkills() walks project > global > bundled (first-wins),
parses SKILL.md frontmatter, no INDEX.json. readBrowserSkill() does
the same for a single name. tombstoneBrowserSkill() moves a skill
into .tombstones/<name>-<ts>/ for recoverability.

Frontmatter parser handles the subset browser-skills need: scalars
(host, description, trusted, version, source), string lists
(triggers), and arg-mapping lists ([{name, description}, ...]).
Quoted values handle colons; trusted defaults to false.

Bundled tier path is auto-detected from the binary install location;
project tier comes from git rev-parse; global is ~/.gstack/. All tier
paths are overridable for hermetic tests.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(browser-skills): \$B skill list/show/run/test/rm subcommands

handleSkillCommand dispatches to per-subcommand handlers; spawnSkill is
the load-bearing function that:

  1. Mints a per-spawn scoped token (read+write only) bound to the
     skill name + spawn-id.
  2. Builds the spawn env:
     - trusted: passes process.env minus GSTACK_TOKEN (defense in depth).
     - untrusted: minimal allowlist (LANG, LC_ALL, TERM, TZ) + locked
       PATH; explicitly drops anything matching TOKEN/KEY/SECRET/etc.
       Also drops AWS_/AZURE_/GCP_/GOOGLE_APPLICATION_/ANTHROPIC_/OPENAI_/
       GITHUB_/GH_/SSH_/GPG_/NPM_TOKEN/PYPI_ patterns.
   3. Always injects GSTACK_PORT + GSTACK_SKILL_TOKEN last (cannot be
     overridden by parent env).
  4. Spawns bun run script.ts -- <args> with cwd=skillDir, captures
     stdout (1MB cap), stderr, and timeout-kills past the deadline.
  5. Revokes the token in finally{}, always.

list output prints the resolved tier inline so "why did it run that
one?" never becomes a debugging mystery (Codex finding #4 mitigation).

server.ts threads the listen port to meta-commands via MetaCommandOpts.daemonPort.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(browser-skills): bundled hackernews-frontpage reference skill

Smallest interesting browser-skill: scrapes HN front page, returns
30 stories as JSON. No auth, stable HTML, fully fixture-tested.

Files:
  SKILL.md                          frontmatter + prose
  script.ts                         exports parseStoriesFromHtml(html)
                                    main: goto + html + parse + JSON.stringify
  _lib/browse-client.ts             vendored copy of the SDK
  fixtures/hn-2026-04-26.html       captured front page (5 stories)
  script.test.ts                    13 assertions against the fixture

The parser is a pure function over HTML so script.test.ts runs
without a daemon (just imports parseStoriesFromHtml and asserts).

This exercises every Phase 1 component end-to-end:
  - browse-client SDK (script imports browse from ./_lib/)
  - 3-tier lookup (hackernews-frontpage lives in the bundled tier)
  - scoped tokens (read+write is enough for goto + html)
  - spawn lifecycle (\$B skill run hackernews-frontpage)
  - file-fixture testing (\$B skill test hackernews-frontpage)

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(skill-validation): cover bundled browser-skills

Adds 7 assertions per bundled skill at <root>/browser-skills/<name>/:
  - SKILL.md exists
  - frontmatter parses with required fields (name/host/triggers/args)
  - script.ts exists
  - _lib/browse-client.ts exists and matches the canonical SDK byte-for-byte
  - script.test.ts exists
  - script.ts imports browse from ./_lib/browse-client

The byte-identical SDK check enforces the version-pinning contract:
when the canonical SDK at browse/src/browse-client.ts changes, every
bundled skill's _lib/ copy must be re-synced or this test fails.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* docs(designs): add BROWSER_SKILLS_V1 design doc

Captures the 13 locked decisions, two-axis trust model (daemon-side
scoped tokens + process-side env access), 3-tier lookup, file
layout, and full responses to all 8 Codex outside-voice findings.
Includes Phase 2-4 sketches for future branches.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* docs(todos): replace self-authoring-\$B P1 with browser-skills phases

Phase 1 of the browser-skills design shipped on this branch (sidesteps
the in-daemon isolation problem the original P1 was blocked on). The
new entries enumerate the work that remains:

  P1: Phase 2 (/scrape + /automate skill templates)
  P2: Phase 3 (resolver injection at session start)
  P2: Phase 4 (eval infra + fixture staleness + OS sandbox)

Cross-references docs/designs/BROWSER_SKILLS_V1.md for the full
architecture and the 8 Codex review findings + responses.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* release: v1.9.0.0 — browser-skills runtime

VERSION 1.8.0.0 → 1.9.0.0. CHANGELOG entry leads with what humans
can do today (hand-write deterministic browser scripts, run them in
200ms via \$B skill run). Notes explicitly that agent authoring
lands in next release; no fabricated perf numbers.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(browser-skills-e2e): exercise dispatch with bundled hackernews-frontpage

Covers the full \$B skill list/show/test pipeline against the real
bundled reference skill (defaultTierPaths picks up <repo>/browser-skills/).
Verifies frontmatter shape, the three-tier walk surfaces the bundled
entry, and \$B skill test successfully runs the bundled script.test.ts
in a child bun process.

\$B skill run end-to-end against the live network is intentionally NOT
covered here (would be flaky against news.ycombinator.com); the spawn
lifecycle is exercised in browser-skill-commands.test.ts using inline
synthetic skills.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* docs: regen SKILL.md to surface the skill META command

bun run gen:skill-docs picked up the new \`skill\` command from
COMMAND_DESCRIPTIONS in browse/src/commands.ts.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* release: bump v1.9.0.0 → v1.13.0.0

Main shipped through v1.11.1.0 while this branch was in flight; v1.12.x
is presumed claimed by another in-flight branch. Use v1.13.0.0 as the
next available slot.

Updated VERSION, package.json, and the CHANGELOG header. Entry body
unchanged.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* release: bump v1.13.0.0 → v1.16.0.0

Main shipped v1.13.0.0 (claude outside-voice skill), v1.14.0.0
(sidebar REPL), and v1.15.0.0 (slim preamble + plan-mode E2E)
while this branch was in flight. Use v1.16.0.0 as the next
available slot.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(browse-skills): atomic write helper for /skillify (D3)

stageSkill writes a candidate skill into ~/.gstack/.tmp/skillify-<spawnId>/
with restrictive perms. commitSkill does an atomic fs.renameSync into the
final tier path with realpath/lstat discipline (refuses symlinked staging
dirs, refuses to clobber existing skills). discardStaged is the cleanup
path for test failures and approval rejections, idempotent and bounded
to the per-spawn wrapper. validateSkillName enforces lowercase/digits/
dashes only, no path-escape characters.

Implements the D3 contract from the v1.19.0.0 plan review: never a
half-written skill on disk. Test fail or approval reject = rm -rf the
temp dir, no tombstone for never-approved skills.

Closes Codex finding #5 (atomic skill packaging) for Phase 2a.

34 unit assertions covering: stage validation, file-path escape rejection,
permission check, atomic rename, clobber refusal, symlink refusal, project
tier unresolved, idempotent discard, end-to-end happy + simulated test
failure + approval reject paths.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(scrape): /scrape <intent> skill template

One entry point for pulling page data. Three paths under the hood:

1. Match — agent reads $B skill list, semantically matches the user's
   intent against each skill's triggers + description + host. Confident
   match = $B skill run <name> in ~200ms.
2. Prototype — no match, drive the page with $B goto/text/html/links etc.
   Return JSON, append a one-line "say /skillify" nudge.
3. Mutating refusal — verbs like submit/click/fill route to /automate
   (Phase 2b P0); /scrape is read-only by contract.

Match decision lives in the agent, not the daemon. No new code in
browse/src/, no expanded daemon command surface, no new prompt-injection
blast radius.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* feat(skillify): /skillify codifies last /scrape into permanent skill

The productivity multiplier. /scrape discovers the flow; /skillify writes
it as deterministic Playwright-via-browse-client code so the next /scrape
on the same intent runs in ~200ms.

11-step flow with three locked contracts from the v1.19.0.0 plan review:

D1 — Provenance guard. Walk back ≤10 agent turns for a clearly-bounded
/scrape result. Refuse with one specific message if cold. No silent
synthesis from chat fragments.

D2 — Synthesis input slice. Extract ONLY the final-attempt $B calls that
produced the JSON the user accepted, plus the user's intent string. Drop
failed selectors, drop unrelated chat, drop earlier-session content.
Closes Codex finding #6 by picking option (b) from the design doc:
re-prompt from agent's own context, not a structured recorder.

D3 — Atomic write. Stage to ~/.gstack/.tmp/skillify-<spawnId>/, run
$B skill test against the temp dir, only rename into the final tier path
on test pass + user approval. Test fail or approval reject = rm -rf the
temp dir entirely.

Default tier: global (~/.gstack/browser-skills/<name>/). --project flag
overrides to per-project. Generated test must include at least one ★★
assertion (parsed JSON has expected shape + non-empty key fields), not a
smoke ★ assertion.

Bun runtime distribution (Codex finding #7) carries over to Phase 4.
Documented in the skill's Limits section.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(browser-skills): gate-tier E2E for /scrape + /skillify (D4)

Five scenarios cover the productivity loop and the contracts locked
during the v1.19.0.0 plan review:

  scrape-match-path           — intent matching bundled hackernews-frontpage
                                routes via $B skill run, no prototype phase
  scrape-prototype-path       — no matching skill, drives $B against a local
                                file:// fixture, returns JSON, suggests
                                /skillify
  skillify-happy-path         — /scrape then /skillify; skill written to
                                ~/.gstack/browser-skills/<name>/ with the
                                full file tree; SKILL.md prose body must
                                not contain conversation fragments (D2)
  skillify-provenance-refusal — cold /skillify with no prior /scrape refuses
                                with the D1 message; nothing on disk (D1)
  skillify-approval-reject    — /scrape then /skillify but reject in the
                                approval gate; temp dir is removed, nothing
                                at the final tier path (D3)

All five gate-tier (~$0.50-$1.50 each, ~$5 total per CI run). Set EVALS=1
to enable. Uses local file:// fixtures so prototype + skillify scenarios
run deterministically without network.

Touchfiles registers all 5 entries with proper deps on scrape/**,
skillify/**, browse/src/browser-skill-write.ts, and the Phase 1 runtime
modules. The match-path test depends on the bundled hackernews-frontpage
skill so its touchfile includes browser-skills/hackernews-frontpage/**.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* docs(browser-skills): TODOS Phase 2a + design doc D1-D4 decisions

TODOS.md:
- Narrows existing P1 (was "/scrape and /automate") to "/scrape and
  /skillify" — the /scrape + /skillify wedge ships in this branch.
  Codex finding #6 (synthesis) removed from Cons (resolved by D2);
  finding #7 (Bun runtime) stays as the open carry-over.
- Adds new ## P0 above PACING_UPDATES_V0 for the /automate follow-up.
  Same skillify pattern as /scrape, different trust profile (per-step
  confirmation gate when running non-codified). Reuses /skillify and
  the D3 helper as-is. Effort M.

BROWSER_SKILLS_V1.md:
- Phase table re-organized into 1, 2a, 2b, 3, 4. Phase 1 + Phase 2a
  consolidate into v1.19.0.0 ship (the v1.16.0.0 branch-internal
  bump never landed on main).
- New "Phase 2a" sub-section captures the four decisions locked
  during /plan-eng-review:
    D1 — provenance guard (≤10 turn walk-back, refuse if cold)
    D2 — synthesis input slice (final-attempt $B calls only,
         closes Codex finding #6)
    D3 — atomic write discipline (temp-dir-then-rename via new
         browse/src/browser-skill-write.ts helper)
    D4 — full test scope (5 gate E2E + 1 unit + smoke)
- New "Phase 2b" sketch for /automate: same skillify machinery,
  per-mutating-step confirmation gate, deferred to next branch.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* release: v1.16.0.0 -> v1.19.0.0 — browser-skills Phase 1 + 2a

Consolidates the v1.16.0.0 branch-internal bump (Phase 1 runtime, never
landed on main) with Phase 2a (/scrape + /skillify + atomic-write helper)
into one v1.19.0.0 ship per CLAUDE.md "Never orphan branch-internal
versions" rule.

Headline: Browser-skills land end-to-end. /scrape <intent> first call
drives the page; second call runs the codified script in 200ms.

The unified CHANGELOG entry covers:
- Phase 1 runtime: $B skill list/show/run/test/rm, scoped tokens,
  3-tier storage, bundled hackernews-frontpage reference.
- Phase 2a: /scrape + /skillify gstack skills, browser-skill-write.ts
  atomic helper, 5 gate-tier E2E + 34 unit assertions.

Numbers table updated: 5 new modules (+browser-skill-write), 2 new
gstack skills, 6 of 8 Codex outside-voice findings resolved (synthesis
#6 closed by D2; Bun runtime #7 + OS sandbox #1 stay deferred to Phase 4).

/automate (Phase 2b) is split out as P0 in TODOS for the next branch.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* fix(commands): tighten descriptions for LLM-judge baseline pinning

The skill-llm-eval test "baseline score pinning" failed CI on three
retry attempts: judge gave command_reference.actionability=3, baseline
demands ≥4. Judge cited 8 specific gaps in COMMAND_DESCRIPTIONS.

This commit closes 7 of 8 by tightening the descriptions:

- press: documents that key names are case-sensitive Playwright keys,
  shows modifier syntax (Shift+Enter, Control+A), links the full key
  list. Removes the "is this case-sensitive?" guesswork.
- is: documents that <sel> accepts either a CSS selector OR an @ref
  token from a prior snapshot, and that property values are case-
  sensitive.
- scroll: documents that there is no --by/--to amount option, points
  at `js window.scrollTo(0, N)` for pixel-precise scrolling.
- js / eval: clarifies that both run in the same JS sandbox, the
  difference is just inline expr (js) vs file (eval).
- storage: clarifies sessionStorage is read-only via this command,
  points at `js sessionStorage.setItem(...)` for the write path.
- chain: walks through how to invoke (pipe a JSON array of arrays to
  $B chain), confirms it stops at the first error.
- cdp: explains how to discover allowed methods (read cdp-allowlist.ts)
  + shows a concrete example invocation.
- domain-skill: explains that the "classifier flag" is set automatically
  by the L4 prompt-injection scan (agents do not set it manually);
  enumerates the full lifecycle verbs.

The 8th gap (storage set syntax conflict) is also resolved as part of
the storage rewrite.

Two pipe-character bugs caught by the existing
`no command description contains pipe character` guard at
`test/gen-skill-docs.test.ts:595`: the chain example originally used
`echo '[...]' | $B chain` (literal pipe) and the cdp description used
`tab|browser` / `trusted|untrusted` (also literal pipes). Both rewritten
to keep markdown table cells intact.

Verification: 696/0 pass on skill-validation + gen-skill-docs after
regen across all hosts. The CI llm-judge eval will re-run against the
new SKILL.md and should hit actionability ≥4 reliably.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* docs(browser): rewrite BROWSER.md as complete reference

Full rewrite covering the gstack browser surface as of v1.19.0.0. Up from
488 to 1,299 lines, 26 top-level sections.

Adds previously-undocumented subsystems:

- The productivity loop: /scrape + /skillify with D1 (provenance guard),
  D2 (final-attempt-only synthesis), D3 (atomic-write discipline) contracts.
- Browser-skills runtime: anatomy, three-tier storage, scoped tokens, trust
  model (capability + env axes), sibling SDK distribution, atomic-write
  helper, bundled hackernews-frontpage reference.
- Domain-skills: per-site agent notes with quarantined → active → global
  state machine and the L4-classifier auto-promotion gate.
- Pair-agent: dual-listener architecture, 26-command tunnel allowlist,
  canDispatchOverTunnel pure gate, three token types (root, setup key,
  scoped), denial log path + salt model.
- Security stack L1-L6: layer table, thresholds (BLOCK/WARN/LOG_ONLY/
  SOLO_CONTENT_BLOCK), ensemble rule, classifier model paths, env knobs.
- Side Panel deep dive: Terminal pane (Claude PTY) as the primary surface
  with Activity/Refs/Inspector as debug overlays, WS auth via
  Sec-WebSocket-Protocol, gstackInjectToTerminal cross-pane plumbing.
- CDP escape hatch: $B cdp deny-default allowlist, $B inspect CSS inspector,
  $B ux-audit page structure extraction.
- Meta commands previously undocumented: tabs/frames/state/watch/inbox/
  tab-each, with usage and storage paths.
- Authentication: three token types with lifetimes, SSE session cookie,
  PTY session cookie, token registry behavior.
- Full source map: 30+ file inventory of browse/src/ vs the old 11-file
  list.

Preserves from before: architecture diagram, daemon lifecycle, snapshot
ref staleness, screenshot modes, goto file:// vs load-html semantics,
batch endpoint, JS await wrapping, env vars, performance numbers vs MCP,
Playwright acknowledgments, dev guide.

Cross-links to ARCHITECTURE.md, CLAUDE.md, docs/REMOTE_BROWSER_ACCESS.md,
docs/designs/BROWSER_SKILLS_V1.md, scrape/SKILL.md, skillify/SKILL.md,
TODOS.md so anyone landing on BROWSER.md can navigate to the load-bearing
companion docs.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* fix(server): tab-ownership gate keys on tabPolicy, not isWrite

Browser-skill spawns hit `403: Tab not owned by your agent` on every
first run because the gate at server.ts:639 fired for any non-root
write, regardless of the token's tabPolicy. The bundled
hackernews-frontpage reference skill failed identically. Every
/skillify-generated skill failed identically. The user's natural
tabs have no claimed owner — by design — so any skill driving
them via `goto` (a write) was 403'd.

The intent in skill-token.ts:79 was always correct: `tabPolicy: 'shared'`
with the comment "skill scripts may switch tabs as needed." The
enforcement just ignored it.

Two surgical changes:

browser-manager.ts:checkTabAccess — gate now keys on options.ownOnly
only. Shared-policy tokens (skill spawns, default scoped clients) get
permissive access — root-equivalent for the tab gate. Own-only tokens
(pair-agent over the ngrok tunnel) still require ownership for every
read and write. isWrite stays in the signature for callers that want
to log or branch elsewhere; it no longer gates the decision.

server.ts:639 — gate predicate narrowed from
  (WRITE_COMMANDS.has(command) || tokenInfo.tabPolicy === 'own-only')
to just
  tokenInfo.tabPolicy === 'own-only'
The 'newtab' exemption stays. Shared tokens skip the gate entirely;
own-only tokens still hit it. Comment block above the gate updated to
document the new predicate intent.

Pair-agent isolation is intact. Tunnel tokens still default to
tabPolicy: 'own-only', still must `newtab` first to get a tab they
can drive, still can't dispatch any of the 23 commands outside the
tunnel allowlist.

The capability gate (scope checks) and rate limits already constrain
what local scoped clients can do; tab ownership was never a security
boundary for them — only for pair-agent. This release makes the
enforcement match the original design intent.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* test(server): lock the shared-vs-own-only tab gate contract

The pre-fix tests at tab-isolation.test.ts:43,57 encoded the broken
behavior as the contract — they specifically asserted "scoped agent
cannot write to unowned tab," which was the exact failure mode that
broke browser-skills. They passed because they tested the wrong
invariant.

This commit replaces those tests with explicit shared-vs-own-only
coverage that documents what each policy actually means:

- Shared scoped agents (skill spawns, default scoped clients) can
  read AND write any tab — unowned, their own, or another agent's.
  The capability is gated by scope checks + rate limits, not by tab
  ownership.
- Own-only scoped agents (pair-agent over tunnel) cannot read OR
  write any tab they don't own. Pre-fix this case was conflated with
  shared writes; now it's explicit.

9 unit assertions on checkTabAccess, up from 6. Each test names
the policy axis it's covering so a future refactor can't quietly
flip the contract.

Adds source-shape regression test 10a in server-auth.test.ts:
"tab gate predicate is own-only-scoped, not write-scoped." The
gate's `if (...)` line MUST contain `tabPolicy === 'own-only'` and
MUST NOT contain `WRITE_COMMANDS.has(command) ||`. If a future
refactor re-introduces the write-scoped gate, this fails immediately
in free-tier `bun test`.

Updates the marker for the existing newtab-excluded test to match
the new comment block ("Tab ownership check (own-only tokens /
pair-agent isolation)").

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* release: v1.19.0.0 -> v1.20.0.0 — fix tab-ownership footgun

Patch release on top of v1.19.0.0. The shipping headline of v1.19.0.0
(/scrape + /skillify productivity loop) was broken on first run in any
session where the daemon already had a tab. Bundled
hackernews-frontpage failed identically. Every /skillify-generated
skill failed identically.

The fix narrows the tab-ownership gate from "any non-root write" to
"tabPolicy === 'own-only' only." Pair-agent isolation (the v1.6.0.0
threat model) is intact; local skill spawns get their original
behavior back.

VERSION: 1.19.0.0 -> 1.20.0.0
package.json version: synced.

CHANGELOG entry leads with the user-visible impact: the productivity
loop works again, no half-second-stalls of confused 403s. Includes
before/after metrics on the bundled reference skill and the broken-
contract pre-fix tests that hid the regression.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* docs(claude): sharpen CHANGELOG rule — diff between main and ship

Codifies what was already implicit in the existing "Never orphan
branch-internal versions" + "Only document what shipped between main
and this change" sections, but with sharper language and concrete
NEVER examples.

The rule: a CHANGELOG entry is the diff between main and the shipping
branch — what users get when they upgrade. NOT how the branch got
there. Branch-internal version bumps, mid-branch bug fixes, plan
review outcomes, and patch narratives all belong in PR descriptions
and commit messages, not in CHANGELOG.

Adds explicit examples of phrasing to NEVER use:
  - "v1.X had a bug that v1.Y fixes" (mentions a branch-internal version)
  - "The shipping headline of v1.X was broken because..." (apologizes
    for never-released state)
  - "Pre-fix tests encoded the broken behavior" (contributor's victory
    lap, not user benefit)
  - "Two surgical edits, both in the dispatch path" (micro-narrative
    of the patch)

The constructive replacement: describe the released system as a
property, not as a fix. "Browser-skills run end-to-end with the
expected tab-access semantics." If a property is worth calling out,
document it in the trust-model section, not as a "we fixed X" callout.

Pairs with feedback_no_shame_changelog and
feedback_changelog_harden_against_critics memories — entries should
read as a flex even to a hostile screenshotter, never admit prior
breakage.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* docs(changelog): consolidate v1.20.0.0 as the diff vs main

Rewrites the v1.20.0.0 entry to describe what users get when they
upgrade from main (v1.17.0.0) to this release: browser-skills
end-to-end. Drops all branch-internal narrative — Phase 1 / Phase 2a
labels, the v1.8.0.0 P1 history paragraph, the test-counts-by-phase
split, and the patch micro-narrative for the tab-policy semantics.

The previously-separate v1.19.0.0 entry (a branch-internal version
that never landed on main) collapses into v1.20.0.0 per the
"Never orphan branch-internal versions" rule.

Tab-access policies are now documented as a property of the trust
model: `'shared'` (skill spawns) is permissive, `'own-only'`
(pair-agent over the tunnel) is strict. No "fix" framing, no
mention of an intermediate state where it was broken.

Adds the BROWSER.md rewrite and the new tab-isolation +
server-auth source-shape regression tests to the itemized changes.

The reverse-chronological order remains: v1.20.0.0 → v1.17.0.0 →
v1.16.0.0 → v1.15.0.0 → ... Gaps (v1.18, v1.19) are fine — those
were branch-internal version numbers that never landed.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

---------

Co-authored-by: Claude Opus 4.7 <noreply@anthropic.com>
2026-04-28 20:08:04 -07:00

56 KiB
Raw Blame History

Browser — Complete Reference

gstack's browser surface in one document. Headless Chromium daemon, ~70+ commands, ref-based element selection, codifiable browser-skills, real-browser mode with a Chrome side panel, an in-sidebar Claude PTY, an ngrok pair-agent flow, and a layered prompt-injection defense — all behind a compiled CLI that prints plain text to stdout. ~100-200ms per call. Zero context-token overhead.

If you've used gstack in the last release or two, the productivity loop is the new headline: /scrape <intent> drives a page once, /skillify codifies the flow into a deterministic Playwright script, and the next /scrape on the same intent runs in ~200ms instead of ~30 seconds of agent re-exploration.


Quick start

# One-time: build the binary (browse/dist/browse, ~58MB)
bun install && bun run build

# Set $B once and forget about it
B=./browse/dist/browse           # or ~/.claude/skills/gstack/browse/dist/browse

# Drive a page
$B goto https://news.ycombinator.com
$B snapshot -i                   # @e refs you can click/fill/inspect later
$B click @e30                    # click ref 30 from the snapshot
$B text                          # get clean page text
$B screenshot /tmp/hn.png

# Codify a repeated flow
/scrape latest hacker news stories
/skillify                        # writes ~/.gstack/browser-skills/hn-front/...
/scrape hacker news front page   # second call: 200ms via the codified skill

# Watch Claude work in real time
$B connect                       # headed Chromium + Side Panel extension

Table of contents

  1. What it is
  2. The productivity loop — /scrape + /skillify
  3. Architecture
  4. Command reference
  5. Snapshot system + ref-based selection
  6. Browser-skills runtime
  7. Domain-skills (per-site agent notes)
  8. Real-browser mode ($B connect)
  9. Side Panel + sidebar agent
  10. Pair-agent — remote agents over an ngrok tunnel
  11. Authentication + tokens
  12. Prompt-injection security stack (L1L6)
  13. Screenshots, PDFs, visual inspection
  14. Local HTML — goto file:// vs load-html
  15. Batch endpoint
  16. Console, network, dialog capture
  17. JS execution — js + eval
  18. Tabs, frames, state, watch, inbox
  19. CDP escape hatch + CSS inspector
  20. Performance + scale
  21. Multi-workspace isolation
  22. Environment variables
  23. Source map
  24. Development + testing
  25. Cross-references
  26. Acknowledgments

What it is

A compiled CLI binary that talks to a persistent local Chromium daemon over HTTP. The CLI is a thin client — it reads a state file, sends a command, prints the response to stdout. The daemon does the real work via Playwright.

Everything that was a Chrome MCP server in the early days now happens through plain stdout. No JSON-schema framing, no protocol negotiation, no persistent WebSocket — Claude's Bash tool already exists, so we use it.

Three escalating modes:

  • Headless (default). Daemon runs Chromium with no visible window. Fastest, cheapest, what skills like /qa, /design-review, /benchmark use by default.
  • Headed via $B connect. Same daemon, but Chromium is visible (rebranded as "GStack Browser") with the Side Panel extension auto-loaded. You watch every command tick through in real time.
  • Pair-agent over a tunnel. Daemon binds a second listener that ngrok forwards. A remote agent (Codex, OpenClaw, Hermes, anything that can speak HTTP) drives your local browser through a 26-command allowlist with a scoped, single-use token.

The productivity loop

The shipped headline of v1.19.0.0. Two gstack skills wrap the browser-skills runtime so the second time you ask Claude to scrape a page, it runs in ~200ms.

/scrape <intent>

One entry point for pulling page data. Three paths under the hood:

  1. Match path (~200ms) — agent runs $B skill list, semantically matches the intent against each skill's triggers: array + description + host, and runs $B skill run <name> if a confident match exists.
  2. Prototype path (~30s) — no match, agent drives the page with $B goto, $B text, $B html, $B links, etc., returns the JSON, and appends a one-line "say /skillify" suggestion.
  3. Mutating-intent refusal — verbs like submit, click, fill route to /automate (Phase 2b, P0 in TODOS.md). /scrape is read-only by contract.

/skillify

Codifies the most recent successful /scrape prototype into a permanent browser-skill on disk. Eleven steps, three locked contracts:

  • D1 — Provenance guard. Walks back ≤10 agent turns for a clearly-bounded /scrape result. Refuses with one specific message if cold. No silent synthesis from chat fragments.
  • D2 — Synthesis input slice. Extracts ONLY the final-attempt $B calls that produced the JSON the user accepted, plus the user's intent string. Drops failed selectors, drops chat, drops earlier-session content.
  • D3 — Atomic write. Stages everything to ~/.gstack/.tmp/skillify-<spawnId>/, runs $B skill test against the temp dir, and only renames into the final tier path on test pass + user approval. Test fail or rejection: rm -rf the temp dir entirely. No half-written skill ever appears in $B skill list.

Mutating-flow sibling /automate is split out as P0 in TODOS.md and ships on the next branch — same skillify machinery, per-mutating-step confirmation gate when running non-codified.

See docs/designs/BROWSER_SKILLS_V1.md for the full design + decision trail.


Architecture

┌─────────────────────────────────────────────────────────────────┐
│  Claude Code                                                    │
│                                                                 │
│  $B goto https://staging.myapp.com                              │
│       │                                                         │
│       ▼                                                         │
│  ┌──────────┐    HTTP POST     ┌──────────────┐                 │
│  │ browse   │ ──────────────── │ Bun HTTP     │                 │
│  │ CLI      │  127.0.0.1:rand  │ daemon       │                 │
│  │          │  Bearer token    │              │                 │
│  │ compiled │ ◄──────────────  │  Playwright  │──── Chromium    │
│  │ binary   │  plain text      │  API calls   │    (headless    │
│  └──────────┘                  └──────────────┘     or headed)  │
│   ~1ms startup                  persistent daemon               │
│                                 auto-starts on first call       │
│                                 auto-stops after 30 min idle    │
└─────────────────────────────────────────────────────────────────┘

Daemon lifecycle

  1. First call. CLI checks <project>/.gstack/browse.json for a running server. None found — it spawns bun run browse/src/server.ts in the background. Daemon launches headless Chromium via Playwright, picks a random port (1000060000), generates a bearer token, writes the state file (chmod 600), starts accepting requests. ~3 seconds.
  2. Subsequent calls. CLI reads the state file, sends an HTTP POST with the bearer token, prints the response. ~100-200ms round trip.
  3. Idle shutdown. After 30 minutes of no commands, daemon shuts down and cleans up the state file. Next call restarts it.
  4. Crash recovery. If Chromium crashes, the daemon exits immediately — no self-healing, don't hide failure. CLI detects the dead daemon on the next call and starts a fresh one.

Multi-workspace isolation

Each project root (detected via git rev-parse --show-toplevel) gets its own daemon, port, state file, cookies, and logs. No cross-workspace collisions. State at <project>/.gstack/browse.json.

Workspace State file Port
/code/project-a /code/project-a/.gstack/browse.json random (1000060000)
/code/project-b /code/project-b/.gstack/browse.json random (1000060000)

Command reference

~70 commands across read, write, and meta. Selectors accept CSS, @e refs from snapshot, or @c refs from snapshot -C. Full table:

Reading

Command Description
text [sel] Clean page text (or scoped to a selector)
html [sel] innerHTML, or full page HTML if no selector
links All links as text → href
forms Form fields as JSON
accessibility Full ARIA tree
media [--images|--videos|--audio] [sel] Media elements with URLs, dimensions, types
data [--jsonld|--og|--meta|--twitter] Structured data: JSON-LD, OG, Twitter Cards, meta tags

Inspection

Command Description
js <expr> Run inline JavaScript expression in page context, return as string
eval <file> Run JS from a file (path under /tmp or cwd; same sandbox as js)
css <sel> <prop> Computed CSS value
attrs <sel|@ref> Element attributes as JSON
is <prop> <sel|@ref> State check: visible, hidden, enabled, disabled, checked, editable, focused
console [--clear|--errors] Captured console messages
network [--clear] Captured network requests
dialog [--clear] Captured dialog messages
cookies All cookies as JSON
storage / storage set <key> <val> Read both localStorage + sessionStorage; set localStorage
perf Page load timings
inspect [sel] [--all] [--history] Deep CSS via CDP — full rule cascade, box model, computed styles
ux-audit Page structure for behavioral analysis: site ID, nav, headings, text blocks, interactive elements
cdp <Domain.method> [json-params] Raw CDP method dispatch (deny-default; allowlist in cdp-allowlist.ts)

Navigation

Command Description
goto <url> Navigate to URL (http://, https://, file://)
load-html <file> Load local HTML in memory (no file:// URL; survives viewport scale changes)
back, forward, reload Standard nav
url Current page URL
wait <sel|--networkidle|--load> Wait for element, network idle, or page load (15s timeout)

Interaction

Command Description
click <sel|@ref> Click element
fill <sel> <val> Fill input
select <sel> <val> Select dropdown option (value, label, or visible text)
hover <sel> Hover element
type <text> Type into focused element
press <key> Playwright keyboard key (case-sensitive: Enter, Tab, ArrowUp, Shift+Enter, Control+A, ...)
scroll [sel|@ref] Scroll element into view, or jump to page bottom if no selector
viewport [<WxH>] [--scale <n>] Set viewport size + optional deviceScaleFactor 1-3 (retina screenshots)
upload <sel> <file> [...] Upload file(s)
dialog-accept [text] Auto-accept next alert/confirm/prompt; text is sent for prompts
dialog-dismiss Auto-dismiss next dialog

Style + cleanup

Command Description
style <sel> <prop> <val> Modify CSS property (with undo support)
style --undo [N] Undo last N style changes
cleanup [--ads|--cookies|--sticky|--social|--all] Remove page clutter
prettyscreenshot [--scroll-to <sel|text>] [--cleanup] [--hide <sel>...] [path] Clean screenshot with optional cleanup, scroll, hide

Visual

Command Description
screenshot [--selector <css>] [--viewport] [--clip x,y,w,h] [--base64] [sel|@ref] [path] Five modes: full page, viewport, element crop, region clip, base64
pdf [path] [--format letter|a4|legal] [...] PDF with full layout: format, width/height, margins, header/footer templates, page numbers, --tagged for accessibility, --toc waits for Paged.js
responsive [prefix] Three screenshots: mobile (375x812), tablet (768x1024), desktop (1280x720)
diff <url1> <url2> Text diff between two URLs

Cookies + headers

Command Description
cookie <name>=<value> Set cookie on current page domain
cookie-import <json> Import cookies from JSON file
cookie-import-browser [browser] [--domain d] Import from installed Chromium browsers (interactive picker, or --domain for direct import)
header <name>:<value> Set custom request header (sensitive values auto-redacted)
useragent <string> Set user agent (triggers context recreation, invalidates refs)

Tabs + frames

Command Description
tabs List open tabs
tab <id> Switch to tab
newtab [url] [--json] Open new tab; --json returns {tabId, url} for programmatic use
closetab [id] Close tab
tab-each <command> [args...] Fan out a command across every open tab; returns JSON
frame <sel|@ref|--name n|--url pattern|main> Switch to iframe context (or back to main); clears refs

Extraction

Command Description
download <url|@ref> [path] [--base64] Download URL or media element using browser cookies
scrape <images|videos|media> [--selector] [--dir] [--limit] Bulk download all media from page; writes manifest.json
archive [path] Save complete page as MHTML via CDP

Snapshot

Command Description
snapshot [-i] [-c] [-d N] [-s sel] [-D] [-a] [-o path] [-C] Accessibility tree with @e refs; -i interactive only, -c compact, -d N depth, -s scope, -D diff vs previous, -a annotated screenshot, -C cursor-interactive @c refs

Server lifecycle

Command Description
status Daemon health + mode (headless / headed / cdp)
stop Shut down daemon
restart Restart daemon
connect Launch headed GStack Browser with Side Panel extension
disconnect Close headed Chrome, return to headless
focus [@ref] Bring headed Chrome to foreground (macOS); @ref also scrolls into view
state save|load <name> Save or load browser state (cookies + URLs)

Handoff

Command Description
handoff [reason] Open visible Chrome at current page for user takeover (CAPTCHA, MFA, complex auth)
resume Re-snapshot after user takeover, return control to AI

Meta + chains

Command Description
chain (JSON via stdin) Run a sequence of commands. Pipe [["cmd","arg1",...],...] to $B chain. Stops at first error.
inbox [--clear] List messages from sidebar scout inbox
watch [stop] Passive observation — periodic snapshots while user browses; stop returns summary

Browser-skills runtime

Command Description
skill list List all browser-skills with resolved tier (project > global > bundled)
skill show <name> Print SKILL.md
skill run <name> [--arg k=v...] [--timeout=Ns] Spawn the skill script with a per-spawn scoped token
skill test <name> Run the skill's script.test.ts against bundled fixtures
skill rm <name> [--global] Tombstone a user-tier skill

Domain-skills

Command Description
domain-skill save|list|show|edit|promote-to-global|rollback|rm <host?> Per-site agent notes (host derived from active tab). Lifecycle: quarantined → active (after N=3 successful uses without classifier flag) → global (explicit promote)

Aliases: setcontent, set-content, setContentload-html (canonicalized before scope checks, so a read-scoped token can't use the alias to run a write command).


Snapshot system

The browser's key innovation is ref-based element selection built on Playwright's accessibility tree API. No DOM mutation. No injected scripts. Just Playwright's native AX API.

How @ref works

  1. page.locator(scope).ariaSnapshot() returns a YAML-like accessibility tree.
  2. The snapshot parser assigns refs (@e1, @e2, ...) to each element.
  3. For each ref, it builds a Playwright Locator (using getByRole + nth-child).
  4. The ref→Locator map is stored on BrowserManager.
  5. Later commands like click @e3 look up the Locator and call locator.click().

Ref staleness detection

SPAs can mutate the DOM without navigation (React router, tab switches, modals). When this happens, refs collected from a previous snapshot may point to elements that no longer exist. resolveRef() runs an async count() check before using any ref — if the element count is 0, it throws immediately with a message telling the agent to re-run snapshot. Fails fast (~5ms) instead of waiting for Playwright's 30-second action timeout.

Extended snapshot features

  • --diff (-D). Stores each snapshot as a baseline. On the next -D call, returns a unified diff showing what changed. Use this to verify that an action (click, fill, etc.) actually worked.
  • --annotate (-a). Injects temporary overlay divs at each ref's bounding box, takes a screenshot with ref labels visible, then removes the overlays. Use -o <path> to control the output.
  • --cursor-interactive (-C). Scans for non-ARIA interactive elements (divs with cursor:pointer, onclick, tabindex>=0) using page.evaluate. Assigns @c1, @c2... refs with deterministic nth-child CSS selectors. These are elements the ARIA tree misses but users can still click.

Browser-skills runtime

Per-task directories that codify a repeated browser flow into a deterministic Playwright script. The compounding layer.

Anatomy of a browser-skill

browser-skills/<name>/
├── SKILL.md                        # frontmatter + prose contract
├── script.ts                       # deterministic Playwright-via-browse-client logic
├── _lib/browse-client.ts           # vendored copy of the SDK (~3KB, byte-identical to canonical)
├── fixtures/<host>-<date>.html     # captured page for fixture-replay tests
└── script.test.ts                  # parser tests against the fixture (no daemon required)

The bundled reference is browser-skills/hackernews-frontpage/: scrapes the HN front page, returns 30 stories as JSON. Try it:

$B skill list                            # shows hackernews-frontpage (bundled)
$B skill show hackernews-frontpage
$B skill run hackernews-frontpage        # JSON of 30 stories in ~200ms
$B skill test hackernews-frontpage       # runs script.test.ts against fixture

Three-tier storage

$B skill list walks all three in priority order; first hit wins. Resolved tier is printed inline next to each skill name:

Tier Path When
Project <project>/.gstack/browser-skills/<name>/ Project-specific skills (committed or gitignored)
Global ~/.gstack/browser-skills/<name>/ Per-user skills, all projects
Bundled <gstack-install>/browser-skills/<name>/ Ships with gstack, read-only

Trust model

Two orthogonal axes — daemon-side capability and process-side env — independently configured.

Axis Mechanism Default
Daemon-side capability Per-spawn scoped token bound to read+write scope (browser-driving commands minus admin: eval, js, cookies, storage). Single-use clientId encodes skill name + spawn id. Revoked when spawn exits. Always scoped — never the daemon root token
Process-side env trusted: true frontmatter passes process.env minus GSTACK_TOKEN. trusted: false (default) drops everything except a minimal allowlist (LANG, LC_ALL, TERM, TZ) and pattern-strips secrets (TOKEN/KEY/SECRET/PASSWORD, AWS_, ANTHROPIC_, OPENAI_, GITHUB_, etc.) Untrusted (must opt in)

GSTACK_PORT and GSTACK_SKILL_TOKEN are injected last, so a parent process can't override them.

Output protocol

stdout = JSON. stderr = streaming logs. Exit 0 / non-zero. Default 60s timeout, override via --timeout=Ns. Max stdout 1MB (truncate + non-zero exit if exceeded). Matches gh / kubectl / docker conventions.

How the SDK distribution works

Each skill ships its own copy of browse-client.ts at _lib/browse-client.ts, byte-identical to the canonical browse/src/browse-client.ts. /skillify copies the canonical SDK alongside every generated script. Each skill is fully self-contained: copy the directory anywhere, it runs. Version drift impossible — the SDK is frozen at the version the skill was authored against.

Atomic write discipline (/skillify D3)

browse/src/browser-skill-write.ts provides three primitives:

  • stageSkill(opts) — writes files to ~/.gstack/.tmp/skillify-<spawnId>/<name>/ with restrictive perms.
  • commitSkill(opts) — atomic fs.renameSync into the final tier path. Refuses to follow symlinked staging dirs (lstat check), refuses to clobber existing skills, runs realpath discipline on the tier root.
  • discardStaged(stagedDir)rm -rf the staged dir + per-spawn wrapper. Idempotent. Called on test failure or approval rejection.

There is no "almost shipped" state. Tests pass + user approves = atomic rename. Tests fail or user rejects = staging vanishes.

See docs/designs/BROWSER_SKILLS_V1.md for the full design rationale.


Domain-skills

Different mental model from browser-skills: agent-authored notes about a site (not deterministic scripts). One per hostname. Lifecycle:

  1. domain-skill save <host> — agent writes a note about the site (e.g., "GitHub: PR creation needs --draft flag for non-staff", "X.com: timeline uses cursor pagination, not page numbers"). Default state: quarantined.
  2. After N=3 successful uses without the L4 prompt-injection classifier flagging the note, it auto-promotes to active.
  3. domain-skill promote-to-global <host> lifts it to the global tier (machine-wide, all projects).
  4. domain-skill rollback <host> demotes; domain-skill rm <host> tombstones.

The classifier flag is set automatically by the L4 prompt-injection scan; agents do not set it manually.

Storage:

  • Per-project: <project>/.gstack/domain-skills/<host>.md
  • Global: ~/.gstack/domain-skills/<host>.md

Source: browse/src/domain-skills.ts, domain-skill-commands.ts.


Real-browser mode

$B connect launches GStack Browser — a rebranded Chromium controlled by Playwright with the Side Panel extension auto-loaded and anti-bot stealth patches applied. You watch every command tick through a visible window in real time.

$B connect              # launches GStack Browser, headed
$B goto https://app.com # navigates in the visible window
$B snapshot -i          # refs from the real page
$B click @e3            # clicks in the real window
$B focus                # bring window to foreground (macOS)
$B status               # shows Mode: cdp
$B disconnect           # back to headless mode

The window has a subtle golden shimmer line at the top and a floating "gstack" pill in the bottom-right corner so you always know which Chrome window is being controlled.

What "GStack Browser" means

Not your daily Chrome — a Playwright-managed Chromium with custom branding in the Dock and menu bar, anti-bot stealth (sites like Google and NYTimes work without captchas), a custom user agent, and the gstack extension pre-loaded via launchPersistentContext. Your regular Chrome with your tabs and bookmarks stays untouched.

When to use headed mode

  • QA testing where you want to watch Claude click through your app
  • Design review where you need to see exactly what Claude sees
  • Debugging where headless behavior differs from real Chrome
  • Demos where you're sharing your screen
  • Pair-agent sessions (the remote agent drives your local browser)

CDP-aware skills

When in real-browser mode, /qa and /design-review automatically skip cookie import prompts and headless workarounds — the headed browser already has whatever session you logged into.


Side Panel + sidebar agent

The Chrome extension that ships baked into GStack Browser shows a live activity feed of every browse command in a Side Panel, plus @ref overlays on the page, plus an interactive Claude PTY inside the sidebar.

The Terminal pane (the headline)

The Side Panel's primary surface is the Terminal pane — a live claude -p PTY you can type into directly from the sidebar. Activity / Refs / Inspector are debug overlays behind the footer's debug toggle. WebSocket auth uses Sec-WebSocket-Protocol (browsers can't set Authorization on a WebSocket upgrade), and the PTY session token is a 30-minute HttpOnly cookie minted via POST /pty-session.

The toolbar's Cleanup button and the Inspector's "Send to Code" action both pipe text into the live Claude PTY via window.gstackInjectToTerminal(text), exposed by sidepanel-terminal.js. There's no separate /sidebar-command POST — the live REPL is the only execution surface.

Activity feed

A scrolling feed of every browse command — name, args, duration, status, errors. Shows up in real time as Claude works. Backed by SSE (/activity/stream) that accepts the Bearer token OR the HttpOnly gstack_sse session cookie (30-minute stream-scope cookie minted via POST /sse-session).

Refs tab

After $B snapshot, shows the current @ref list (role + name) so you can see what Claude is targeting.

CSS Inspector

Powered by $B inspect (CDP-based). Click any element on the page to see the full CSS rule cascade, computed styles, box model, and modification history. The "Send to Code" button injects a description into the Claude PTY.

Sidebar architecture

Component Where it lives Notes
Side Panel UI extension/sidepanel.js, sidepanel-terminal.js Chrome extension surface
Background SW extension/background.js Manages tab events, port management
Content script extension/content.js Page overlays, gstack pill
Terminal agent browse/src/terminal-agent.ts PTY spawn, lifecycle, auth
Sidebar utilities browse/src/sidebar-utils.ts URL sanitization, helpers

Before modifying any of these, read the comment block in CLAUDE.md under "Sidebar architecture" — silent failures here usually trace to not understanding the cross-component flow.

Manual install (for your regular Chrome)

If you want the extension in your everyday Chrome (not the Playwright-controlled one):

bin/gstack-extension    # opens chrome://extensions, copies path to clipboard

Or do it manually: chrome://extensions → toggle Developer mode → Load unpacked → navigate to ~/.claude/skills/gstack/extension → pin the extension → enter the port from $B status.


Pair-agent

Remote AI agents (Codex, OpenClaw, Hermes, anything that speaks HTTP) can drive your local browser through an ngrok tunnel. The whole flow is gated by a 26-command allowlist, scoped tokens, and a denial log.

How it works

/pair-agent                     # generates a setup key, prints connection instructions
# Copy the instructions to the remote agent
# Remote agent runs:
#   POST <tunnel-url>/connect with setup key → gets a scoped token (24h, single client)
#   POST <tunnel-url>/command with token → runs allowed commands

Dual-listener architecture (v1.6.0.0+)

When pair-agent activates, the daemon binds two HTTP listeners:

  • Local listener (127.0.0.1:LOCAL_PORT). Full command surface. Never forwarded by ngrok. Used by your Claude Code, the Side Panel, anything on your machine.
  • Tunnel listener (127.0.0.1:TUNNEL_PORT). Locked allowlist — /connect, /command (scoped tokens + 26-command browser-driving allowlist), /sidebar-chat. ngrok forwards only this port.

Root tokens sent over the tunnel return 403. SSE endpoints use a 30-minute HttpOnly gstack_sse cookie (never valid against /command).

The 26-command tunnel allowlist

Defined in browse/src/server.ts as TUNNEL_COMMANDS. Pure gate function canDispatchOverTunnel(command) is exported for unit testing. Set:

goto, click, text, screenshot, html, links, forms, accessibility,
attrs, media, data, scroll, press, type, select, wait, eval,
newtab, tabs, back, forward, reload, snapshot, fill, url, closetab

Notably absent: pair, unpair, cookies, setup, launch, restart, stop, tunnel-start, token-mint, state, connect, disconnect. A remote agent that tries them gets a 403 plus a fresh entry in the denial log.

Tunnel denial log

~/.gstack/security/attempts.jsonl — append-only, salted SHA-256 of source

  • domain only (no raw IP, no full request body), rotates at 10MB with 5 generations. Per-device salt at ~/.gstack/security/device-salt (mode 0600).

See docs/REMOTE_BROWSER_ACCESS.md for the full operator guide.

Tab ownership

Scoped tokens default to tabPolicy: 'own-only'. A paired agent can newtab to create its own tab and drive that tab freely, but it can't goto, fill, or click on tabs another caller owns. tabs lists ALL tab metadata (an accepted tradeoff — see ARCHITECTURE.md), but text/html/snapshot content of unowned tabs is blocked by ownership checks.


Authentication

Three token types, three lifetimes, three scopes.

Token Generated by Lifetime Scope
Root token Daemon startup (random UUID) Daemon process lifetime Full command surface, local listener only — 403 over tunnel
Setup key POST /pair 5 minutes, one-time use Single redemption: present at /connect, get a scoped token
Scoped token POST /connect (with setup key) 24 hours Per-client, allowlist-bound, optionally tab-scoped

The root token is written to <project>/.gstack/browse.json with chmod 600. Every command that mutates browser state must include Authorization: Bearer <token>.

SSE endpoints (/activity/stream, /inspector/events) accept the Bearer token OR a 30-minute HttpOnly gstack_sse cookie minted via POST /sse-session. The ?token=<ROOT> query-param auth is no longer supported. This is what lets the Chrome extension subscribe to the activity feed without putting the root token in extension storage.

The Terminal pane uses a separate session cookie, gstack_pty, minted via POST /pty-session. Different scope — can spawn / drive the live claude PTY, can't dispatch arbitrary /command calls. /health endpoint MUST NOT surface this token.

Token registry

browse/src/token-registry.ts handles mint/validate/revoke for all three types, plus per-token rate limiting. Setup keys are single-use; scoped tokens have a sliding 24h window; the root token is rotated on each daemon startup.


Security stack

Layered defense against prompt injection. Every layer runs synchronously on every user message and every tool output that could carry untrusted content (Read, Glob, Grep, WebFetch, page text from $B).

Layer Module Lives in
L1 Datamarking content-security.ts both server + sidebar agent
L2 Hidden-element strip content-security.ts both
L3 ARIA + URL blocklist + envelope wrapping content-security.ts both
L4 TestSavantAI ML classifier (22MB ONNX) security-classifier.ts sidebar-agent only*
L4b Claude Haiku transcript check security-classifier.ts sidebar-agent only
L5 Canary token (session-exfil detection) security.ts both — inject in compiled, check in agent
L6 combineVerdict ensemble security.ts both

* security-classifier.ts cannot be imported from the compiled browse binary — @huggingface/transformers v4 requires onnxruntime-node which fails to dlopen from Bun compile's temp extract dir. The compiled binary runs L1L3, L5, L6 only.

Thresholds

  • BLOCK: 0.85 — single-layer score that would cause BLOCK if cross-confirmed
  • WARN: 0.75 — cross-confirm threshold. When L4 AND L4b both >= 0.75 → BLOCK
  • LOG_ONLY: 0.40 — gates transcript classifier (skip Haiku when all layers < 0.40)
  • SOLO_CONTENT_BLOCK: 0.92 — single-layer threshold for label-less content classifiers

Ensemble rule

BLOCK only when the ML content classifier AND the transcript classifier both report >= WARN. Single-layer high confidence degrades to WARN — this is the Stack Overflow instruction-writing FP mitigation. Canary leak always BLOCKs (deterministic).

Env knobs

  • GSTACK_SECURITY_OFF=1 — emergency kill switch. Classifier stays off even if warmed. Canary is still injected; just the ML scan is skipped.
  • GSTACK_SECURITY_ENSEMBLE=deberta — opt-in DeBERTa-v3 ensemble. Adds ProtectAI DeBERTa-v3-base-injection-onnx as L4c classifier. 721MB first-run download. With ensemble enabled, BLOCK requires 2-of-3 ML classifiers agreeing at >= WARN.
  • Classifier model cache: ~/.gstack/models/testsavant-small/ (112MB, first run only) plus ~/.gstack/models/deberta-v3-injection/ (721MB, only when ensemble enabled).
  • Attack log: ~/.gstack/security/attempts.jsonl (salted SHA-256 + domain only, rotates at 10MB, 5 generations).
  • Per-device salt: ~/.gstack/security/device-salt (0600).
  • Session state: ~/.gstack/security/session-state.json (cross-process, atomic).

A shield icon in the sidebar header shows the live status. See ARCHITECTURE.md § "Prompt injection defense" for the full threat model.


Screenshots, PDFs, visual

Screenshot modes

Mode Syntax Playwright API
Full page (default) screenshot [path] page.screenshot({ fullPage: true })
Viewport only screenshot --viewport [path] page.screenshot({ fullPage: false })
Element crop (flag) screenshot --selector <css> [path] locator.screenshot()
Element crop (positional) screenshot "#sel" [path] or screenshot @e3 [path] locator.screenshot()
Region clip screenshot --clip x,y,w,h [path] page.screenshot({ clip })

Element crop accepts CSS selectors (.class, #id, [attr]) or @e/@c refs. Tag selectors like button aren't caught by the positional heuristic — use the --selector flag form.

--base64 returns data:image/png;base64,... instead of writing to disk — composes with --selector, --clip, --viewport.

Mutual exclusion: --clip + selector, --viewport + --clip, and --selector + positional selector all throw.

Retina screenshots — viewport --scale

viewport --scale <n> sets Playwright's deviceScaleFactor (context-level, 13 cap):

$B viewport 480x600 --scale 2
$B load-html /tmp/card.html
$B screenshot /tmp/card.png --selector .card
# .card at 400x200 CSS pixels → card.png is 800x400 pixels

--scale N alone (no WxH) keeps the current viewport size. Scale changes trigger a context recreation, which invalidates @e/@c refs — rerun snapshot after. HTML loaded via load-html survives the recreation via in-memory replay. Rejected in headed mode (real browser controls scale).

PDF generation

pdf accepts the full Playwright surface plus a few additions:

  • Layout: --format letter|a4|legal, --width <dim>, --height <dim>, --margins <dim>, --margin-top/right/bottom/left <dim>
  • Structure: --toc (waits for Paged.js if loaded), --outline, --tagged (PDF/A accessibility), --print-background, --prefer-css-page-size
  • Branding: --header-template <html>, --footer-template <html>, --page-numbers
  • Tabs: --tab-id <N> to render a specific tab
  • Large payloads: --from-file <payload.json> (avoids shell argv limits)

Responsive screenshots

responsive [prefix] — three screenshots in one call: mobile (375x812), tablet (768x1024), desktop (1280x720). Saves as {prefix}-mobile.png etc.

prettyscreenshot

Combines cleanup + scroll + element hide in one call:

$B prettyscreenshot --cleanup --scroll-to "hero section" --hide ".cookie-banner" /tmp/clean.png

Local HTML

Two ways to render HTML that isn't on a web server:

Approach When URL after Relative assets
goto file://<abs-path> File already on disk file:///... Resolve against file's directory
goto file://./<rel>, goto file://~/<rel> Smart-parsed to absolute file:///... Same
load-html <file> HTML generated in memory, no parent-dir context needed about:blank Broken (self-contained HTML only)

Both are scoped to files under cwd or $TMPDIR via the same safe-dirs policy as eval. file:// URLs preserve query strings and fragments (SPA routes work).

load-html has an extension allowlist (.html, .htm, .xhtml, .svg) and a magic-byte sniff to reject binary files mis-renamed as HTML. 50MB size cap (override via GSTACK_BROWSE_MAX_HTML_BYTES).

load-html content survives later viewport --scale calls via in-memory replay (TabSession tracks the loaded HTML + waitUntil). The replay is purely in-memory — HTML is never persisted to disk via state save to avoid leaking secrets or customer data.


Batch endpoint

POST /batch sends multiple commands in a single HTTP request. Eliminates per-command round-trip latency — critical for remote agents over ngrok where each HTTP call costs 2-5s.

POST /batch
Authorization: Bearer <token>

{
  "commands": [
    {"command": "text", "tabId": 1},
    {"command": "text", "tabId": 2},
    {"command": "snapshot", "args": ["-i"], "tabId": 3},
    {"command": "click", "args": ["@e5"], "tabId": 4}
  ]
}

Each command routes through handleCommandInternal — full security pipeline (scope checks, domain validation, tab ownership, content wrapping) enforced per command. Per-command error isolation: one failure doesn't abort the batch. Max 50 commands per batch. Nested batches rejected. Rate limiting: 1 batch = 1 request against the per-agent limit.

Pattern: agent crawling 20 pages opens 20 tabs (individual newtab or batch), then POST /batch with 20 text commands → 20 page contents in ~2-3 seconds total vs ~40-100 seconds serial.


Capture

Console, network, and dialog events flow into O(1) circular buffers (50,000 capacity each), flushed to disk asynchronously via Bun.write():

  • Console: .gstack/browse-console.log
  • Network: .gstack/browse-network.log
  • Dialog: .gstack/browse-dialog.log

The console, network, and dialog commands read from the in-memory buffers (not disk) so capture is real-time even when disk is slow.

Dialogs (alert, confirm, prompt) are auto-accepted by default to prevent browser lockup. dialog-accept <text> controls prompt response text.


JS execution

js runs an inline expression. eval runs a JS file. Both run in the same JS sandbox — the only difference is inline-vs-file. Both support await — expressions containing await are auto-wrapped in an async context:

$B js "await fetch('/api/data').then(r => r.json())"   # auto-wrapped
$B js "document.title"                                  # no wrap needed
$B eval my-script.js                                    # file with await

For eval files, single-line files return the expression value directly. Multi-line files need explicit return when using await. Comments containing the literal token "await" don't trigger wrapping.

Path safety: eval rejects paths outside cwd or /tmp. js doesn't read files at all.


Tabs, frames, state

Tabs

$B tabs                          # list all open tabs
$B tab 3                         # switch to tab 3
$B newtab https://example.com    # open new tab, switch to it
$B newtab --json                 # programmatic: returns {"tabId":N,"url":...}
$B closetab                      # close current
$B closetab 2                    # close tab 2
$B tab-each "text"               # run "text" on every tab, return JSON

tab-each <command> fans out a command across every open tab and returns a JSON array — handy for "give me the text of every tab I have open."

Frames

$B frame "#stripe-iframe"        # switch to iframe by selector
$B frame @e7                     # by ref
$B frame --name "checkout"       # by name attribute
$B frame --url "stripe.com"      # by URL pattern match
$B frame main                    # back to top frame

Refs are cleared on switch (the iframe has its own AX tree).

State save/load

$B state save my-session         # save cookies + URLs to .gstack/browse-state-my-session.json
$B state load my-session         # restore

In-memory load-html content is intentionally NOT persisted (avoid leaking secrets to disk).

Watch

$B watch                         # passive observation: snapshot every 5s while user browses
$B watch stop                    # return summary of what changed

Useful when you're driving the browser manually and want Claude to see what you did at the end without spamming snapshot calls.

Inbox

$B inbox                         # list messages from sidebar scout
$B inbox --clear                 # clear after reading

The sidebar scout (a background process the Chrome extension can spawn) drops notes for Claude when the user surfaces something they want noticed. Stored in .gstack/browser-scout.jsonl.


CDP

$B cdp — raw Chrome DevTools Protocol dispatch

Deny-default. Only methods enumerated in browse/src/cdp-allowlist.ts (CDP_ALLOWLIST const) are reachable; any other method returns 403. Each allowlist entry declares scope (tab vs browser) and output (trusted vs untrusted). Untrusted methods (data-exfil-shaped, e.g. Network.getResponseBody) get UNTRUSTED-envelope wrapped output.

$B cdp Page.getLayoutMetrics
$B cdp Network.enable
$B cdp Accessibility.getFullAXTree --json '{"max_depth":5}'

To discover allowed methods: read browse/src/cdp-allowlist.ts.

$B inspect — CDP-based CSS inspector

$B inspect ".header"                # full rule cascade for the header
$B inspect ".header" --all          # include user-agent rules
$B inspect ".header" --history      # show modification history

Returns the matched rule cascade with specificity, computed styles, the box model, and (with --history) every CSS modification made via $B style since the page loaded. Powered by a persistent CDP session per page in browse/src/cdp-inspector.ts.

$B ux-audit

$B ux-audit

Returns JSON with site identity, navigation, headings (capped 50), text blocks, interactive elements (capped 200) — page structure for behavioral analysis without dumping the full HTML. Used by /qa and /design-review for cheap coverage maps.


Performance

Tool First call Subsequent calls Context overhead per call
Chrome MCP ~5s ~2-5s ~2000 tokens (schema + protocol)
Playwright MCP ~3s ~1-3s ~1500 tokens (schema + protocol)
gstack browse ~3s ~100-200ms 0 tokens (plain text stdout)
gstack browse + codified skill ~3s ~200ms 0 tokens (single skill invocation)

In a 20-command browser session, MCP tools burn 30,00040,000 tokens on protocol framing alone. gstack burns zero. The codified-skill path takes a 20-command session down to a single $B skill run call.

Why CLI over MCP

MCP works well for remote services. For local browser automation it adds pure overhead:

  • Context bloat — every MCP call includes full JSON schemas. A simple "get the page text" costs 10x more context tokens than it should.
  • Connection fragility — persistent WebSocket/stdio connections drop and fail to reconnect.
  • Unnecessary abstraction — Claude already has a Bash tool. A CLI that prints to stdout is the simplest possible interface.

gstack skips all of this. Compiled binary. Plain text in, plain text out. No protocol. No schema. No connection management.


Multi-workspace

Each project root (detected via git rev-parse --show-toplevel) gets its own daemon, port, state file, cookies, and logs. No cross-workspace collisions.

Workspace State file Port
/code/project-a /code/project-a/.gstack/browse.json random (1000060000)
/code/project-b /code/project-b/.gstack/browse.json random (1000060000)

Browser-skills three-tier lookup walks project → global → bundled, so a project-tier skill at /code/project-a/.gstack/browser-skills/foo/ shadows the global ~/.gstack/browser-skills/foo/ only inside project-a.


Environment variables

Variable Default Description
BROWSE_PORT 0 (random 1000060000) Fixed port for the HTTP server (debug override)
BROWSE_IDLE_TIMEOUT 1800000 (30 min) Idle shutdown timeout in ms
BROWSE_STATE_FILE .gstack/browse.json Path to state file
BROWSE_SERVER_SCRIPT auto-detected Path to server.ts
BROWSE_CDP_URL (none) Set to channel:chrome for real-browser mode
BROWSE_CDP_PORT 0 CDP port (used internally)
BROWSE_HEADLESS_SKIP 0 Skip Chromium launch entirely (test harness only)
BROWSE_TUNNEL 0 Activate the dual-listener tunnel architecture (requires NGROK_AUTHTOKEN)
BROWSE_TUNNEL_LOCAL_ONLY 0 Test-only — bind both listeners locally without ngrok
GSTACK_BROWSE_MAX_HTML_BYTES 52428800 (50MB) load-html size cap
GSTACK_SECURITY_OFF unset Emergency kill switch — disable ML classifier
GSTACK_SECURITY_ENSEMBLE unset Set to deberta for 3-classifier ensemble (721MB download)

Source map

browse/
├── src/
│   ├── cli.ts                   # Thin client — reads state, sends HTTP, prints
│   ├── server.ts                # Bun HTTP daemon — routes commands, dual-listener
│   ├── browser-manager.ts       # Chromium lifecycle, tabs, ref map, crash detection
│   ├── browse-client.ts         # Canonical SDK — what skills import as _lib/browse-client.ts
│   ├── snapshot.ts              # AX tree → @e/@c refs → Locator map; -D/-a/-C handling
│   ├── read-commands.ts         # Non-mutating: text, html, links, js, css, is, dialog, ...
│   ├── write-commands.ts        # Mutating: goto, click, fill, upload, dialog-accept, ...
│   ├── meta-commands.ts         # state, watch, inbox, frame, ux-audit, chain, diff, ...
│   ├── browser-skills.ts        # 3-tier walk + frontmatter parser + tombstones
│   ├── browser-skill-commands.ts # $B skill list/show/run/test/rm + spawnSkill
│   ├── browser-skill-write.ts   # D3 atomic stage/commit/discard helper for /skillify
│   ├── skill-token.ts           # mintSkillToken / revokeSkillToken (per-spawn, scoped)
│   ├── domain-skills.ts         # Per-site agent notes (state machine: quarantined→active→global)
│   ├── domain-skill-commands.ts # $B domain-skill save/list/show/edit/promote/rollback/rm
│   ├── cdp-allowlist.ts         # Deny-default CDP method allowlist
│   ├── cdp-bridge.ts            # CDP session lifecycle bridge
│   ├── cdp-commands.ts          # $B cdp dispatcher
│   ├── cdp-inspector.ts         # $B inspect — persistent CDP session per page
│   ├── activity.ts              # ActivityEntry, CircularBuffer, SSE subscribers, privacy filtering
│   ├── buffers.ts               # Console/network/dialog circular buffers (O(1) ring)
│   ├── tab-session.ts           # Per-tab session state (load-html replay, ref map scope)
│   ├── token-registry.ts        # Mint/validate/revoke for root + setup keys + scoped tokens
│   ├── sse-session-cookie.ts    # 30-min HttpOnly cookie for /activity/stream + /inspector/events
│   ├── pty-session-cookie.ts    # Separate scope: live Claude PTY auth
│   ├── tunnel-denial-log.ts     # ~/.gstack/security/attempts.jsonl writer (salted)
│   ├── path-security.ts         # validateOutputPath / validateReadPath / validateTempPath
│   ├── url-validation.ts        # URL safety checks for goto
│   ├── content-security.ts      # L1-L3: datamarking, hidden strip, ARIA, URL blocklist, envelopes
│   ├── security.ts              # L5 canary + L6 verdict combiner + thresholds
│   ├── security-classifier.ts   # L4 ML classifier (TestSavant + optional DeBERTa ensemble)
│   ├── terminal-agent.ts        # Side Panel Claude PTY manager (auth + lifecycle)
│   ├── sidebar-utils.ts         # Sidebar URL sanitization + helpers
│   ├── cookie-import-browser.ts # Decrypt + import cookies from real Chromium browsers
│   ├── cookie-picker-routes.ts  # HTTP routes for /cookie-picker/*
│   ├── cookie-picker-ui.ts      # Self-contained HTML/CSS/JS for cookie picker
│   ├── network-capture.ts       # Network request capture for $B network
│   ├── media-extract.ts         # Media element extraction for $B media
│   ├── project-slug.ts          # Project slug derivation for state paths
│   ├── error-handling.ts        # safeUnlink / safeKill / isProcessAlive
│   ├── platform.ts              # OS detection (macOS, Linux, Windows)
│   ├── telemetry.ts             # Anonymous opt-in usage telemetry
│   ├── find-browse.ts           # Locate running daemon or bootstrap
│   └── config.ts                # Config resolution (env / files)
├── test/                        # Integration tests + HTML fixtures
└── dist/
    └── browse                   # Compiled binary (~58MB, Bun --compile)

browser-skills/
└── hackernews-frontpage/        # Bundled reference skill
    ├── SKILL.md
    ├── script.ts
    ├── _lib/browse-client.ts
    ├── fixtures/hn-2026-04-26.html
    └── script.test.ts

scrape/SKILL.md.tmpl             # /scrape gstack skill — match-or-prototype entry point
skillify/SKILL.md.tmpl           # /skillify gstack skill — codify last /scrape into permanent skill

Development

Prerequisites

  • Bun v1.0+
  • Playwright's Chromium (installed automatically by bun install)

Quick start

bun install                      # install deps + Playwright Chromium
bun test                         # all integration tests (~3s for browse-only)
bun run dev <cmd>                # run CLI from source (no compile)
bun run build                    # compile to browse/dist/browse

Dev mode vs compiled binary

During development, use bun run dev instead of the compiled binary. It runs browse/src/cli.ts directly with Bun, so you get instant feedback:

bun run dev goto https://example.com
bun run dev text
bun run dev snapshot -i
bun run dev click @e3

The compiled binary (bun run build) is only needed for distribution. It produces a single ~58MB executable at browse/dist/browse using Bun's --compile flag.

Running tests

bun test                                    # all tests
bun test browse/test/commands               # command integration tests
bun test browse/test/snapshot               # snapshot tests
bun test browse/test/cookie-import-browser  # cookie import unit tests
bun test browse/test/browser-skill-write    # D3 atomic-write helper tests
bun test browse/test/tunnel-gate-unit       # canDispatchOverTunnel pure tests

Tests spin up a local HTTP server (browse/test/test-server.ts) serving HTML fixtures from browse/test/fixtures/, then exercise the CLI against those pages.

Adding a new command

  1. Add the handler in read-commands.ts (non-mutating) or write-commands.ts (mutating), or meta-commands.ts (server / lifecycle).
  2. Register the route in server.ts.
  3. Add the entry to COMMAND_DESCRIPTIONS in browse/src/commands.ts (with a clear description and usage — the gen-skill-docs validation suite enforces no | characters in description).
  4. Add a test case in browse/test/commands.test.ts with an HTML fixture if needed.
  5. Run bun test to verify.
  6. Run bun run build to compile.
  7. Run bun run gen:skill-docs to regenerate SKILL.md (the command appears in the command-reference table downstream).

Adding a new browser-skill

For a hand-written skill: copy browser-skills/hackernews-frontpage/, update SKILL.md frontmatter, rewrite script.ts against your target site, re-capture the fixture, update the parser test. bun test validates the SKILL.md contract (sibling SDK byte-identity, frontmatter schema).

For an agent-written skill: drive the page once with /scrape <intent>, say /skillify, accept the proposed name in the approval gate. The skill lands at ~/.gstack/browser-skills/<name>/ after the test passes.

Deploying to the active skill

The active skill lives at ~/.claude/skills/gstack/. After making changes:

cd ~/.claude/skills/gstack
git fetch origin && git reset --hard origin/main
bun run build

Or copy the binary directly:

cp browse/dist/browse ~/.claude/skills/gstack/browse/dist/browse

Cross-references

  • ARCHITECTURE.md — system-level architecture, dual-listener tunnel design, prompt-injection defense threat model
  • CLAUDE.md — project-level instructions, sidebar architecture notes, security-stack constraints
  • docs/REMOTE_BROWSER_ACCESS.md — operator guide for /pair-agent (setup keys, scoped tokens, denial log)
  • docs/designs/BROWSER_SKILLS_V1.md — design doc for browser-skills runtime (Phase 1 + 2a + roadmap)
  • scrape/SKILL.md/scrape skill: match-or-prototype data extraction
  • skillify/SKILL.md/skillify skill: codify last /scrape into permanent skill
  • TODOS.md/automate (Phase 2b P0), Phase 3 resolver injection, Phase 4 eval + sandbox

Acknowledgments

The browser automation layer is built on Playwright by Microsoft. Playwright's accessibility tree API, locator system, and headless Chromium management are what make ref-based interaction possible. The snapshot system — assigning @ref labels to AX tree nodes and mapping them back to Playwright Locators — is built entirely on top of Playwright's primitives. Thank you to the Playwright team for building such a solid foundation.

The prompt-injection L4 layer uses TestSavantAI/distilbert-v1.1-32 (112MB ONNX), and the optional ensemble layer uses ProtectAI/deberta-v3-base-prompt-injection-v2 (721MB ONNX) — both run locally via @huggingface/transformers.

The CDP escape hatch is gated by an allowlist directly inspired by Codex's T2 outside-voice review during the v1.4 design pass: deny-default with an explicit allowlist, not allow-default with a denylist.