Add `homelab browser run|open` so agents can drive the cluster's headful Chrome (chrome-service) over CDP from the devvm. The headless playwright/mcp browser can load anti-bot sites and fill their forms, but the gated submit silently fails — e.g. the Stirling Ackroyd Fixflo tenant portal returned net::ERR_FILE_NOT_FOUND on its pre-submit check and hung, creating nothing. Driving the real headful Chrome submits first try. That capability already existed but was undiscoverable, so it cost ~40 min + redundant form re-runs to find; now it is one command, versioned, test-covered, and `browser --help` carries the when-to-use signature + an error-code cheat-sheet so the right tool is reached at the right moment (the failure was judgment, not setup). - port-forward svc/chrome-service:9222 (tunnels API-server->pod, so it bypasses the :9222 NetworkPolicy), assert non-headless via /json/version, connect_over_cdp, inject the same vendored stealth.js the in-cluster callers use; the port-forward is always torn down, on success and on error. - node CDP client pinned to playwright-core@1.48.2 to match the v1.48.0-noble image (Chromium 130); self-provisioned lazily into ~/.cache/homelab, no per-user setup. - default is a fresh incognito context (safe for the shared browser + concurrent callers); --shared-context reuses the warmed persistent profile. - TDD: cmd_browser_test.go covers arg parsing, headless detection, the version pin, the help cheat-sheet, and a stealth.js drift guard. Verified end-to-end against bot.sannysoft.com (real Chrome UA, webdriver hidden, plugins/WebGL spoofed) and `browser open`. - docs: README v0.8 section, ADR-0013, and a chrome-service.md "driving from outside the cluster" section. Closes: code-nepg Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
4.9 KiB
homelab browser verbs: headful (anti-bot) web automation via cluster Chrome
v0.8 adds browser run, browser open, and browser --help. They package a
capability that already existed but was undiscoverable: driving the cluster's
headful Chrome (chrome-service — real Chrome under Xvfb, CDP on
svc/chrome-service:9222) from the devvm, for sites that detect and block
headless automation.
Motivating incident (2026-06-22)
Logging a washing-machine repair on the Stirling Ackroyd Fixflo tenant
portal: the headless @playwright/mcp browser loaded the site and filled the
entire multi-step form, but the final submit silently failed — Fixflo's
pre-submit POST /IssuePreCreationCheck returned net::ERR_FILE_NOT_FOUND, the
spinner hung, no issue was created. Root cause = headless-Chrome detection. The
fix was to drive the headful chrome-service over connect_over_cdp — it
submitted first try (Fixflo ref IS22657587). That capability was documented
(docs/architecture/chrome-service.md) but not packaged or discoverable, so
it took ~40 min, three redundant full form re-runs, and a user hint. The agent
also misread ERR_FILE_NOT_FOUND as "network egress" and retried blind instead
of inspecting the network panel.
Decisions
- Mechanics in
homelab, not a~/.claudeskill. A standalone skill was rejected: the CLI is run every session (so the verb is discoverable), is versioned, multi-user, and test-covered. A private, untested skill is none of those. The command owns only the deterministic mechanics (port-forward, stealth injection, lifecycle) — the agent supplies the Playwright script, so judgment stays out of the CLI (the founding rule, ADR-0004/0005). - The failure was judgment, not setup friction, so the CLI is paired with a
one-line pointer in always-in-context
~/code/CLAUDE.mdand a diagnostic payload inbrowser --help: the when-to-use signature (a site loads but a gated action fails/hangs, or one request 500s/aborts while siblings 200 → suspect headless detection) and an error-code cheat-sheet (ERR_FILE_NOT_FOUND= request resolved/intercepted by the automation layer, not egress; egress failures areERR_CONNECTION_REFUSED/_TIMED_OUT/_NAME_NOT_RESOLVEDand would break the page load too). A command the agent doesn't think to run is useless; the cheat-sheet is the actual fix for the misdiagnosis. - Reach the pod via
kubectl port-forward, thenconnect_over_cdpto localhost. port-forward tunnels API-server→pod, so it bypasses the:9222NetworkPolicy that gates in-cluster callers — the devvm needs no namespace label. Readiness is asserted against/json/version: the endpoint must report a realChrome/…, neverHeadlessChrome(the whole point). The forward is always torn down (process-group kill + signal handler), on success and on error — an acceptance requirement. - Default to a fresh incognito context;
--shared-contextopts into the warmed profile. chrome-service is a single shared browser with a persistent profile. A fresh, always-closed context is safe for concurrent callers (tripit's fare scrape connects per-quote) and is what production already does. The warmed persistent profile (cookies from a manual noVNC login) is opt-in for flows that need a pre-logged-in session. - Pin the node CDP client to
playwright-core@1.48.2to match the chrome-service image minor (mcr.microsoft.com/playwright:v1.48.0-noble, Chromium 130).connect_over_cdpspeaks the browser's CDP, and protocol changes between Playwright minors — the devvm's ambient Python Playwright was 1.58, a 10-minor skew. The pin makes behaviour deterministic across the fleet regardless of local drift.playwright-core(notplaywright) because no browser binary is needed — we connect to the remote one. - Self-provision the client lazily, no per-user setup. The pinned client is
installed once into
~/.cache/homelab/browser-client/(idempotent, version- guarded) on first use, alongside the embedded runner + stealth files. node is already fleet-wide; this avoids coupling the feature to a provisioner change and keeps it self-contained and self-healing. The client runs on the devvm, sosetInputFilesstreams local files to the remote browser over CDP — nochmod/staging-dir workaround on the CDP path. - Vendor
stealth.js, guard against drift. The CLI embeds a byte-for-byte copy ofstacks/chrome-service/files/stealth.js(the source of truth the in-cluster callers use) viago:embed; a unit test fails if the copy drifts.go:embedcan't reach outside the package dir, hence the vendored copy rather than a path reference. - Scope held at two action verbs + help.
run(arbitrary script — the workhorse) andopen(navigate + title/text/screenshot — a quick check) cover the surface. Both are write-tier; the barebrowser/--helpis read. Re-measure viausage top(ADR-0011) before adding more.