name: kernel-browser-harness
description: Best practices for using browser-use's open-source browser-harness with Kernel cloud browsers over CDP. Use when driving a Kernel browser from browser-harness, extracting a CDP URL from kernel browsers create, or running multi-step or parallel harness sessions against Kernel.
Browser-Harness with Kernel Cloud Browsers
This skill documents how to drive a Kernel cloud browser with browser-harness (browser-use's open-source CDP harness). browser-harness already supports any CDP endpoint via the BU_CDP_WS env var — the work this skill covers is extracting that URL from the Kernel CLI, picking the right kernel browsers create flags, and using the Kernel session ID as the harness daemon name so sessions don't collide.
When to Use This Skill
Use this skill when you need to:
- Drive a Kernel browser from
browser-harnessinstead of the user's local Chrome - Extract a CDP URL from
kernel browsers create -o jsonto feed intoBU_CDP_WS - Span multiple harness calls across a long task (mint, drive, inspect, drive more, tear down)
- Run parallel harness sessions against multiple Kernel browsers without socket collisions
Prerequisites
- Load the
kernel-cliskill for Kernel CLI installation and authentication. - If
browser-harnessis not already on$PATH, install it per its setup-prompt. If it is already installed, do not reinstall or run setup prompts — go straight to minting the Kernel browser. - Install
jqonly if missing (used to pull fields from the CLI's-o jsonoutput).
Environment Variables
| Variable | Description | Default |
|---|---|---|
KERNEL_API_KEY |
Required unless kernel login was run. Used by the CLI to mint sessions. |
(none) |
BU_CDP_WS |
CDP WebSocket URL browser-harness connects to. Set on the first call only — the daemon caches the connection for subsequent calls under the same BU_NAME. |
(local Chrome discovery) |
BU_NAME |
Namespaces the harness daemon socket. Use the Kernel session ID (BU_NAME=$SESSION_ID) — collision-proof per session, no extra naming convention to remember. |
default |
Basic Usage
Fast path: create exactly one Kernel browser, save the JSON, derive every value from that saved JSON, then pass the exact CDP URL to the first harness call. Do not re-run kernel browsers create just to recover variables or copy/paste values from terminal output.
SESSION=$(kernel browsers create --stealth --timeout 1800 -o json)
printf '%s\n' "$SESSION" > /tmp/kernel-session.json
SESSION_ID=$(jq -r '.session_id' /tmp/kernel-session.json)
CDP_WS=$(jq -r '.cdp_ws_url' /tmp/kernel-session.json)
echo "live view: $(jq -r '.browser_live_view_url // empty' /tmp/kernel-session.json)"
BU_NAME="$SESSION_ID" BU_CDP_WS="$CDP_WS" browser-harness <<'PY'
new_tab("https://news.ycombinator.com")
wait_for_load()
print(page_info())
PY
kernel browsers delete "$SESSION_ID"
Kernel browsers boot headful by default — the create response includes browser_live_view_url. Print it so the user can watch the agent work, or pass --headless to opt out (no live view, smaller image).
kernel browsers create flag picks
Common choices when minting a session for harness use:
| Flag | Use For |
|---|---|
--stealth (-s) |
Bot-detection bypass — on for most public sites |
--headless (-H) |
Opt out of headful default — no live view, no VNC |
--profile-name NAME |
Reuse a saved Kernel profile (logged-in state, cookies, localStorage) |
--proxy-id ID |
Route through a Kernel-managed proxy (residential, regional, etc.) |
--timeout N |
Idle timeout in seconds (default 60, max 259200) — bump for long agent runs |
--start-url URL |
Open a URL when the session boots |
--save-changes |
Persist profile mutations back when the session ends |
See kernel browsers create --help for the full list.
Multi-Step Usage
For tasks that span more than one shell call — mint, drive, inspect, drive more, tear down — the daemon does the work. browser-harness (addressed by BU_NAME) holds the CDP connection between invocations, so BU_CDP_WS only needs to be set on the first call.
With BU_NAME=$SESSION_ID as the convention, every subsequent harness call is just:
BU_NAME=$SESSION_ID browser-harness <<'PY'
print(js("document.title"))
PY
— and parallel sessions are automatic: each Kernel browser has a unique session ID, so two BU_NAME=$SESSION_ID invocations against different sessions never collide on the daemon socket.
If you lose $SESSION_ID across shell calls, recover it with kernel browsers list -o json | jq.
For replay recording around a harness session, see the kernel-cli skill's replays reference.
Common Gotchas
BU_CDP_WS unreachablemid-task: the Kernel session probably hit its idle timeout. Default is 60s — pass--timeout 1800(or whatever fits the task) at create time.CDP URL is a JWT-signed
wss://endpoint — paste it directly intoBU_CDP_WS, no rewriting or stripping.Daemon won't pick up a new session: if you mint a new Kernel browser but reuse a stale
BU_NAME, the daemon stays connected to the old CDP URL. UsingBU_NAME=$SESSION_IDavoids this entirely (new session, new socket). If you do hit it manually,browser-harness --reloadstops the daemon so the next call connects fresh.Extra creates waste time and leak money: one task should usually call
kernel browsers createonce. Save the-o jsonresponse immediately and reuse it; do not run a second create after seeing output, and do not hardcode IDs or CDP URLs from a previous command.Live view URL is for the human: print
browser_live_view_urlfrom the create response so the user can watch. The agent only needscdp_ws_url.Always tear down: run
kernel browsers delete "$SESSION_ID"when the task ends. Sessions bill until idle timeout. If you are wrapping several steps in a script, register a cleanuptrapafterSESSION_IDis known; for ad hoc command use, an explicit delete at the end is clearer. If you lost the SID,kernel browsers list -o json | jqrecovers it.Skill responsibilities:
browser-harness'sSKILL.mdowns helper usage (new_tab,page_info,js, …) and the heredoc form. Thekernel-cliskill ownskernel browsers create / list / get / deleteandreplayslifecycle. This skill only owns the CLI-to-harness wiring.
Quick Reference
# Mint once
SESSION=$(kernel browsers create --stealth --timeout 1800 -o json)
printf '%s\n' "$SESSION" > /tmp/kernel-session.json
SESSION_ID=$(jq -r '.session_id' /tmp/kernel-session.json)
CDP_WS=$(jq -r '.cdp_ws_url' /tmp/kernel-session.json)
# Drive (first call seeds the daemon with the exact signed CDP URL)
BU_NAME="$SESSION_ID" BU_CDP_WS="$CDP_WS" browser-harness <<'PY'
new_tab("https://example.com"); print(page_info())
PY
# Later calls reuse BU_NAME only
BU_NAME="$SESSION_ID" browser-harness <<'PY'
print(js("document.title"))
PY
# Recovery if you lost the SID across shells
kernel browsers list -o json | jq
# Teardown
kernel browsers delete "$SESSION_ID"