Hive hive.browser-automation
Required before any browser_* tool call. Teaches the screenshot + browser_click_coordinate workflow that reaches shadow-DOM inputs selectors can't see, the CSS-pixel coordinate rule (not physical px), rich-text editor quirks ("send button stays disabled" failures), and CSP gotchas. Covers Chrome via CDP through the GCU Beeline extension. Skipping this causes repeated failures on LinkedIn / Reddit / X. Verified against real production sites 2026-04-11.
git clone https://github.com/aden-hive/hive
T=$(mktemp -d) && git clone --depth=1 https://github.com/aden-hive/hive "$T" && mkdir -p ~/.claude/skills && cp -r "$T/core/framework/skills/_default_skills/browser-automation" ~/.claude/skills/aden-hive-hive-hive-browser-automation && rm -rf "$T"
core/framework/skills/_default_skills/browser-automation/SKILL.mdGCU Browser Automation
All GCU browser tools drive a real Chrome instance through the Beeline extension and Chrome DevTools Protocol (CDP). That means clicks, keystrokes, and screenshots are processed by the actual browser's native hit testing, focus, and layout engines — not a synthetic event layer. Understanding this unlocks strategies that make hard sites easy.
Coordinates
Every browser tool that takes or returns coordinates operates in fractions of the viewport (0..1 for both axes). Read a target's proportional position off
browser_screenshot — "this button is about 35% from the left and 20% from the top" → pass (0.35, 0.20). Rect-returning tools (browser_get_rect, browser_shadow_query, and the rect inside focused_element) also return fractions. The tools convert to CSS pixels internally before dispatching to Chrome.
browser_screenshot() → image + cssWidth/cssHeight in meta browser_click_coordinate(x, y) → x, y are fractions 0..1 browser_hover_coordinate(x, y) → fractions browser_press_at(x, y, key) → fractions browser_get_rect(selector) → rect → rect.cx / rect.cy are fractions browser_shadow_query(...) → rect → same
Why fractions: every vision model (Claude ~1.15 MP target, GPT-4o 512-px tiles, Gemini, local VLMs) resizes or tiles images differently before the model sees the pixels. Proportions survive every such transform; pixel coordinates only "work" per-model and silently break when you swap backends. Four-decimal precision (
0.0001 ≈ 0.17 CSS px on a 1717-wide viewport) is more than enough for the tightest targets.
Exception for zoomed elements: pages that use
zoom or transform: scale() on a container (LinkedIn's #interop-outlet, some embedded iframes) render in a scaled local coordinate space. getBoundingClientRect there may not match CDP's hit space. Prefer browser_shadow_query (which handles the math and returns fractions) or visually pick coordinates from a screenshot. Avoid raw browser_evaluate + getBoundingClientRect() for coord lookup — that returns CSS px and will be wrong when fed to click tools.
Screenshot + coordinates is shadow-agnostic — prefer it on shadow-heavy sites
Start with
browser_snapshot when you need to inspect the page structure or find ordinary controls. If the snapshot does not show the thing you need, shows stale or misleading refs, or cannot prove where a visible target is, take browser_screenshot and use the screenshot + coordinate path. This is especially useful on sites that use Shadow DOM heavily
Why:
- CDP hit testing walks shadow roots natively.
routes through Chrome's native hit tester, which traverses open shadow roots automatically. You don't need to know the shadow structure.browser_click_coordinate(x, y) - Keyboard dispatch follows focus into shadow roots. After a click focuses an input (even one three shadow levels deep),
with no selector dispatches keys tobrowser_press(...)
's computed focus target.document.activeElement - Screenshots render the real layout regardless of DOM implementation.
Whereas
wait_for_selector, browser_click(selector=...), browser_type(selector=...) all use document.querySelector under the hood, which stops at shadow boundaries. They cannot see elements inside shadow roots. For shadow-DOM inputs, use browser_type_focused after focusing via click-coordinate.
Recommended workflow on shadow-heavy sites
→ JPEG; meta includesbrowser_screenshot()
/cssWidth
for reference.cssHeight- Identify the target visually → estimate its proportional position
where each is in(fx, fy)
.0..1
→ tool converts to CSS px and dispatches; CDP native hit testing focuses the element. The response includesbrowser_click_coordinate(fx, fy)
— use it to verify you actually focused what you intended.focused_element: {tag, id, role, contenteditable, rect, inFrame?, ...}
is in fractions (same space as your input). When focus is inside a same-origin iframe, the descriptor reports the inner element and addsrect
breadcrumbs.inFrame: [...]
→ inserts text intobrowser_type_focused(text="...")
(traverses into same-origin iframes automatically). Shadow roots, iframes, Lexical, Draft.js, ProseMirror all just work. Usedocument.activeElement
instead when you have a reliable CSS selector for a light-DOM element.browser_type(selector, text)- Verify via
ORbrowser_screenshot
on a known-reachable marker (e.g. check that the Send button'sbrowser_get_attribute
flipped toaria-disabled
).false
The click→type loop (canonical pattern)
- Call
to click the target element.browser_click_coordinate(x, y) - Check the
field in the response — it tells you what actually received focus (tag, id, role, contenteditable, rect).focused_element - If the focused element is editable, call
to insert text. Use tools to verify the text took effect — prefer checking the underlyingbrowser_type_focused(text="...")
/.value
viainnerText
or confirming the submit button enabled. A screenshot alone can mislead: narrow input boxes visually clip long text, so only a portion may appear on screen even though the full string was accepted.browser_evaluate - If it is NOT editable, your click landed on the wrong thing — refine coordinates and retry. Do NOT reach for
+browser_evaluate
or shadow-root traversals. The problem is the click target, not the typing method.execCommand('insertText')
browser_click (selector-based) also returns focused_element, so the same check works whether you clicked by selector or coordinate.
Empirically verified (2026-04-11)
Tested against
https://www.reddit.com/r/programming/ whose search input lives at:
document > reddit-search-large [shadow] > faceplate-search-input#search-input [shadow] > input[name="q"]
Shadow-piercing selectors
When you DO want a selector-based approach and know the shadow structure,
browser_shadow_query and browser_get_rect support >>> shadow-piercing syntax:
browser_shadow_query("reddit-search-large >>> #search-input") browser_get_rect("#interop-outlet >>> #ember37 >>> p")
Returns the element's rect as fractions of the viewport (feed
rect.cx / rect.cy directly to click tools). Remember: browser_type and wait_for_selector do not support >>> — only shadow_query and get_rect do.
Navigation and waiting
The basics
browser_navigate(url, wait_until="load") # "load" | "domcontentloaded" | "networkidle" browser_wait_for_selector("h1", timeout_ms=2000) browser_wait_for_text("Some text", timeout_ms=2000) browser_go_back() browser_go_forward() browser_reload()
All return real URLs and titles. On a fast page
navigate(wait_until="load") returns in sub-second. wait_for_selector and wait_for_text typically resolve in single-digit milliseconds on elements already in the DOM.
Timing expectations (measured against real sites)
| Site | Navigate load time |
|---|---|
| example.com | 100–400 ms |
| wikipedia.org | 200–500 ms |
| reddit.com | 1.5–2 s |
| x.com/twitter | 1.2–1.6 s |
| linkedin.com (logged in) | 4–5 s |
For LinkedIn and other heavy SPAs, rely on
sleep() after navigation to let the page hydrate.
After navigate, always let SPA hydrate
Even after
wait_until="load", React/Vue SPAs often render their real chrome in a second pass. Add await sleep(2) to await sleep(3) before querying for site-specific elements. Otherwise wait_for_selector will fail on elements that do exist moments later.
Reading pages efficiently
- Prefer
overbrowser_snapshot
— returns a compact ~1–5 KB accessibility tree vs 100+ KB of raw HTML.browser_get_text("body") - Interaction tools
,browser_click
,browser_type
,browser_type_focused
, andbrowser_fill
wait 0.5 s for the page to settle after a successful action, then attach a fresh accessibility snapshot under thebrowser_scroll
key of their result. Use it to decide your next action — do NOT callsnapshot
separately after every action. Tune the capture viabrowser_snapshot
:auto_snapshot_mode
(full tree, the default),"default"
(trims unnamed structural nodes),"simple"
(only controls — tightest token footprint), or"interactive"
to skip the capture entirely (useful when batching several interactions and you don't need the intermediate trees). Call"off"
explicitly only when you need a newer view or a different mode than what was auto-captured.browser_snapshot - Complex pages (LinkedIn, Twitter/X, SPAs with virtual scrolling) can have DOMs that don't match what's visually rendered — snapshot refs may be stale, missing, or misaligned with visible layout. Try the available snapshot first; when the target is not present in that snapshot or visual position matters, switch to
to orient yourself.browser_screenshot - Only fall back to
for extracting specific small elements by CSS selector.browser_get_text
Typing and keyboard input
ALWAYS click before typing into rich-text editors
The single most common "looks like it worked but send button stays disabled" failure. If you're typing into a modern editor (X/Twitter's Draft.js compose, LinkedIn's post composer, Reddit's comment box, Gmail compose, Slack, Discord, Notion, Monaco, any
contenteditable), click the input area first with browser_click_coordinate or browser_click(selector) before you type.
Why this is necessary:
- React / Vue controlled components don't trust JS-sourced
. React uses event delegation and watches for native pointer/focus events — a.focus()
dispatched via CDP fires the realclick
/pointerdown
/pointerup
/click
sequence that React listens to, and updates its internal state. A JS-onlyfocus
sets.focus()
but the framework's controlled state doesn't see it.document.activeElement - Draft.js (X/Twitter compose) and Lexical (Gmail, LinkedIn DMs) use contenteditable divs with immutable editor state. They only enter "edit mode" after a real click on the editor surface. Typing at them without clicking routes keys to
or gets silently discarded.document.body - Send/submit buttons are bound to framework state, not DOM state. They're typically
wheredisabled={!hasRealContent}
is computed from React/Vue/Svelte state. The input field can have characters in the DOM but the button stays disabled because the framework never saw a real input event.hasRealContent
The symptom is always the same: you type, the characters appear visually, and the send button doesn't enable. The agent then clicks send anyway, nothing happens, and it thinks the post failed.
Safe "click-then-type-then-verify" pattern
-
Focus the real element via a real click (not JS
). Use.focus()
(orbrowser_get_rect(selector)
for shadow sites) to get coordinates, thenbrowser_shadow_query
. Wait ~0.5 s for the editor to open and focus to settle.browser_click_coordinate(cx, cy) -
Type the text. Use
for light-DOM inputs, orbrowser_type(selector, text)
for shadow-DOM / already-focused inputs. Both use CDPbrowser_type_focused(text=...)
by default, which is the most reliable method for rich editors (Lexical, Draft.js, ProseMirror). Wait ~500 ms for framework state to commit.Input.insertText -
Verify the submit button is enabled before clicking it. Use
to check the button'sbrowser_evaluate
ordisabled
attribute. Do NOT trust that typing worked — always check state.aria-disabledPartial visibility is fine. Small single-line inputs, chat boxes with fixed width, and search fields commonly clip or truncate long text visually — only the tail or head may be shown on screen. Don't treat that as failure. What matters is that the framework accepted the input: the submit button enabled, or
/element.value
read viainnerText
contains the full string. If the visible pixels don't match what you typed but the button is enabled and the underlying value is correct, typing succeeded — proceed.browser_evaluate -
Only click send if the button is enabled. If the button is still disabled, try the recovery dance: click the textarea again, press
, press a space, pressEnd
— this forces React to recomputeBackspace
. Then re-check the button state.hasRealContent
Why browser_type
uses Input.insertText
by default
browser_typeInput.insertTextCDP has a dedicated method —
Input.insertText — for committing text into the focused element as if IME just committed it. It bypasses the keyboard event pipeline entirely and works cleanly on every rich-text editor tested to date: Lexical (LinkedIn DMs, Gmail), Draft.js (X compose), ProseMirror (Reddit), Monaco, and plain contenteditable. Playwright uses this under the hood for keyboard.type() on rich editors.
Per-character
Input.dispatchKeyEvent looks equivalent on paper, but some rich editors listen for beforeinput events with a specific shape and route insertion through their own state machine — the raw keys arrive but never get turned into text. That was the exact failure mode that left LinkedIn's message composer empty (and its Send button disabled) during the 2026-04-11 empirical run.
If you need per-keystroke dispatch (autocomplete testing, code editors, animated typing with
delay_ms), pass use_insert_text=False to fall back to the old keyDown/keyUp path.
Neutralizing beforeunload
draft dialogs
beforeunloadWhen a composer has unsent text and you try to navigate away or close the tab, sites like LinkedIn pop a native "You have an unsent message, leave?" confirm dialog via
window.onbeforeunload. Your automation hangs waiting on the dialog — browser_close_tab and browser_navigate both time out.
Strip the handler via
before navigating:browser_evaluate
browser_evaluate(""" (function(){ window.onbeforeunload = null; window.addEventListener('beforeunload', function(e){ e.stopImmediatePropagation(); }, true); return true; })() """) # Now browser_navigate / close_tab work without hitting a confirm
Always include an equivalent cleanup block in any script that types into a compose UI — without it, a script crash mid-type leaves the tab in an unusable state with the draft modal blocking every subsequent automation call.
Verified site-specific quirks
| Site | Editor | Workaround |
|---|---|---|
| X / Twitter compose | Draft.js | Click first, then type with . First 1-2 chars may be eaten — accept truncation or prepend a throwaway char. Verify has before clicking. |
| LinkedIn messaging | contenteditable (inside shadow root) | Use to find the rect, click-coordinate to focus, then (selector-based can't reach shadow). Send button is . |
| LinkedIn feed post composer | Quill/LinkedIn custom | Click the "Start a post" trigger first, wait 1s for modal, click the textarea, type. |
| Reddit comment/post box | ProseMirror | Click the textarea, wait 0.5s for the toolbar to mount, then type. Submit is inside a shreddit-composer. |
| Gmail compose | Lexical | Click the body first. Gmail has a visible after opening a compose window. |
| Slack message box | contenteditable | Click first, then type. Send is a paper-plane button with . |
| Discord | Slate | Click first. Discord's send is implicit on Enter (no button), so just press Enter after typing. |
| Monaco editors (GitHub code review, CodeSandbox) | Monaco | Click first, type with . Monaco listens for input events on a hidden textarea — requires focus to be on that textarea. |
Plain text into a real input
For plain
<input> and <textarea> elements with no framework wrapper (forms on static sites, simple search bars that pass a selector string straight through), browser_type(selector, text) is sufficient — the bridge's internal focus() call does the right thing. But when in doubt, click first. It's cheap insurance.
browser_type(selector, text)
- Sends
(withkeyDown
,key
,code
fields populated) →text
per character (or a singlekeyUp
by default)Input.insertText - Fires real
/keydown
/keypress
/input
events — frameworks that branch onkeyup
orevent.key
see the right valuesevent.code - Matches what Playwright and Puppeteer send
Works on real
<input>, <textarea>, and contenteditable elements. For shadow-DOM inputs, see the "shadow-heavy sites" section above — browser_type(selector=) can't see past shadow boundaries; use browser_type_focused after click-coordinate focus.
Keyboard shortcuts (Ctrl+A, Shift+Tab, Cmd+Enter)
browser_press("a", modifiers=["ctrl"]) # Ctrl+A — select all browser_press("Backspace") # clear selected text browser_press("Enter", modifiers=["meta"]) # Cmd+Enter (mac) — submit browser_press("Tab", modifiers=["shift"]) # Shift+Tab — reverse focus
Accepted modifier names (case-insensitive):
"alt", "ctrl" / "control", "meta" / "cmd", "shift".
Behind the scenes this dispatches the modifier's own
keyDown first, then the main key with code and windowsVirtualKeyCode populated (so Chrome's shortcut dispatcher recognises it), then releases modifiers in reverse order. Without the code + windowsVirtualKeyCode fields Chrome routes the event to the DOM without firing shortcuts — which is what plain string keys get.
Special keys
Recognized without modifiers:
Enter, Tab, Escape, Backspace, Delete, ArrowUp/Down/Left/Right, Home, End, PageUp, PageDown.
Screenshots
browser_screenshot() # viewport, 800 px wide JPEG browser_screenshot(full_page=True) # full scrollable page (overview only — don't click off a full-page shot) browser_screenshot(selector="#header") # clip to element's rect
Returns a JPEG (quality 75, ~50–120 KB) at 800 px wide. The pixel width is purely a bandwidth choice; all tool coordinates are fractions of the viewport and are invariant to image size. Metadata includes
imageWidth (800), cssWidth, cssHeight (for reference), and physicalScale. The image is annotated with a highlight rectangle/dot showing the last interaction (click, hover, type) if one happened on this tab.
The highlight overlay stays visible on the page for 10 seconds after each interaction, then fades. Before a screenshot is likely, make sure your click / hover / type happens <10 s before the screenshot.
Scrolling
- Use large scroll amounts (~2000) when loading more content — sites like Twitter and LinkedIn have lazy loading for paging.
- The scroll result includes a snapshot automatically — no need to call
separately.browser_snapshot - Never re-navigate to the same URL after scrolling — this resets your scroll position and loses loaded content.
Batching actions
- You can call multiple tools in a single turn — they execute in parallel. ALWAYS batch independent actions together. Examples: fill multiple form fields in one turn, navigate + snapshot in one turn, click + scroll if targeting different elements.
- When batching, set
on all but the last action to avoid redundant snapshots.auto_snapshot=false - Aim for 3–5 tool calls per turn minimum. One tool call per turn is wasteful.
Tab management
Close tabs as soon as you are done with them — not only at the end of the task. After reading or extracting data from a tab, close it immediately.
- Finished reading/extracting from a tab?
browser_close(target_id=...) - Completed a multi-tab workflow?
to clean up all your tabsbrowser_close_finished() - More than 3 tabs open? Stop and close finished ones before opening more
- Popup appeared that you didn't need? Close it immediately
browser_tabs returns an origin field for each tab:
— you opened it; you own it; close it when done"agent"
— opened by a link or script; close after extracting what you need"popup"
or"startup"
— leave these alone unless the task requires it"user"
Never accumulate tabs. Treat every tab you open as a resource you must free.
The bridge automatically evicts per-tab state (
_cdp_attached, _interaction_highlights) when a tab is closed, so you can't leak stale annotations or attached-debugger flags.
Site-specific selectors (verified 2026-04-11)
| Target | Selector |
|---|---|
| Global search input | |
| Own profile link | |
| Messaging overlay | (use shadow_query) |
LinkedIn enforces strict Trusted Types CSP. Any script you inject via
browser_evaluate that uses innerHTML = "<...>" will be silently dropped — the wrapper element gets added but its content is empty, no console error. Always use createElement + appendChild + setAttribute for DOM injection on LinkedIn. style.cssText, textContent, and .value assignments are fine (they don't go through the Trusted Types sink).
Reddit (new reddit / shreddit)
| Target | Selector |
|---|---|
| Search input (shadow) | (rect only; type via click-to-focus) |
| Reddit logo (home) | |
| Subreddit posts | custom elements |
| Create post button | |
Reddit's search input lives two shadow levels deep inside
reddit-search-large > faceplate-search-input. You cannot reach it with browser_type(selector=). The working pattern:
→ rectbrowser_shadow_query("reddit-search-large >>> #search-input")
→ click lands on the real shadow input via native hit testing; input becomes focusedbrowser_click_coordinate(rect.cx, rect.cy)
→ dispatches to focused element viabrowser_type_focused(text="query")Input.insertText- Verify by reading
via.value
walking the shadow pathbrowser_evaluate
X / Twitter
| Target | Selector |
|---|---|
| Main search input | |
| Home nav link | |
| Post text area (compose) | |
| Reply buttons on feed | |
| Post / Tweet submit button | |
| Caret (⋯) menu on a post | |
| Confirmation sheet button | |
X uses Draft.js for the compose text editor, which does NOT accept synthetic input reliably. Working workaround:
browser_type(selector='[data-testid="tweetTextarea_0"]', text="...", delay_ms=20). The delay gives Draft.js time to process each keystroke. The first 1–2 characters may still get eaten — accept minor truncation or prepend a throwaway character. After typing, check [data-testid="tweetButton"] has disabled: false before clicking submit.
After submitting, press Escape to close the composer.
File uploads — use browser_upload
, never click the upload button
browser_uploadClicking an
or the button that triggers one (X's photo button, LinkedIn's attach button, Gmail's paperclip) opens Chrome's native OS file picker. That dialog is rendered by the operating system, NOT the page, so CDP cannot see it, cannot interact with it, and the automation wedges. This is the single most common way to lock up a browser session on any "compose with media" flow.<input type="file">
The only correct pattern: call
browser_upload(selector, file_paths). It uses the CDP DOM.setFileInputFiles method, which sets the files directly on the input element's internal state as if the user had picked them — no OS dialog ever opens.
# WRONG — opens the native file picker, agent gets stuck browser_click_coordinate(photo_button_x, photo_button_y) # ❌ # RIGHT — sets the file programmatically, no dialog browser_upload( selector="input[type='file']", # the underlying file input file_paths=["/absolute/path/to/image.png"], )
Finding the file input. On most modern SPAs the visible "Add photo" / "Attach" button is a styled
<button> or <label>, and the real <input type="file"> is hidden (often display:none or opacity:0, positioned offscreen, wrapped in a <label for="...">, or injected on click). Use browser_evaluate to enumerate ALL file inputs on the page first:
browser_evaluate(""" (function(){ const inputs = Array.from(document.querySelectorAll('input[type="file"]')); return inputs.map(el => ({ name: el.name || '', accept: el.accept || '', multiple: el.multiple, id: el.id || '', inViewport: (() => { const r = el.getBoundingClientRect(); return r.width > 0 && r.height > 0; })(), })); })(); """)
Then pass the most specific selector that uniquely identifies the right input (e.g.
input[type='file'][accept*='image'] for a photo-only upload). browser_upload doesn't care if the input is hidden or offscreen — DOM.setFileInputFiles works on any valid file input node, visible or not.
X / LinkedIn / Twitter pattern. On X (
x.com/compose/post), the photo upload input is input[data-testid='fileInput'] — hidden, reachable via browser_upload. On LinkedIn feed compose, look for input[type='file'][accept*='image'] inside the post-creation modal after clicking "Add media" (clicking the Add-media button reveals the input but does NOT open the dialog; only clicking the SECOND layer — the "From computer" entry — would trigger the picker. Stop at the first layer, find the input, call browser_upload).
Verification after upload.
DOM.setFileInputFiles dispatches a change event on the input but NOT the click / focus events that some sites gate their UI on. Always verify the upload actually took effect by screenshotting the composer (the uploaded image should appear as a preview) or by checking for a "preview" / "remove" element that only exists post-upload. If verification fails, the site may be reading the file via some other bridge — fall back to reading the file bytes and pasting them via the clipboard (navigator.clipboard.write with a ClipboardItem) through browser_evaluate.
If a native file picker DOES open (you clicked the wrong thing): there is no recovery via CDP. Press Escape via
browser_press("Escape") immediately — this dismisses the OS dialog in Chrome on Linux/macOS. Then find the actual <input type='file'> and use browser_upload.
Common pitfalls
- Typing into a rich-text editor without clicking first → send button stays disabled. Draft.js (X), Lexical (Gmail, LinkedIn DMs), ProseMirror (Reddit), and React-controlled
elements only register input as "real" when the element received a native focus event — JS-sourcedcontenteditable
is not enough..focus()
now does this automatically via a real CDP pointer click before inserting text, but always verify the submit button'sbrowser_type
state before clicking send. See the "ALWAYS click before typing" section above.disabled - Using per-character
on Lexical / Draft.js editors → keys dispatch but text never appears. Those editors interceptkeyDown
and route insertion through their own state machine; raw keyDown events are silently dropped.beforeinput
now usesbrowser_type
by default (the CDP IME-commit method) which these editors accept cleanly. Only setInput.insertText
when you explicitly need per-keystroke dispatch.use_insert_text=False - Leaving a composer with text then trying to navigate →
dialog hangs the bridge. LinkedIn and several other sites pop a native "unsent message" confirm.beforeunload
andbrowser_navigate
both time out against this. Always stripclose_tab
viawindow.onbeforeunload = null
before any navigation after typing in a composer, or wrap your logic in abrowser_evaluate
that runs the cleanup block.try/finally - Click landed in the wrong region (sidebar / header instead of target). Check
in the click response — it's ground truth for what actually got focused, including thefocused_element
breadcrumb when focus ends up inside a same-origin iframe. If it isn't the target (e.g.inFrame
when you meant to hit a composer), adjust the fraction and retry. Coordinates you pass are fractions of the viewport; the tool multiplies byclassName: "msg-conversation-listitem__link"
/cssWidth
internally, so a wrong result means your estimated proportion was off — not that any scale went sideways.cssHeight - Accidentally passing pixels to click / hover / press_at. The tools reject any coord outside
with a clear error. If you see that error, you passed a pixel (like 815) instead of a fraction (like 0.475). Use[-0.1, 1.5]
to get exact fractional cx/cy, or read proportions offbrowser_get_rect
.browser_screenshot - Calling
on a shadow element. It'll always time out. Usewait_for_selector
or the screenshot + coordinate strategy.browser_shadow_query - Relying on
in injected scripts on LinkedIn. Silently discarded. UseinnerHTML
+createElement
.appendChild - Not waiting for SPA hydration.
fires before React/Vue rendering on many sites. Add a 2–3 s sleep before querying for chrome elements.wait_until="load" - Using
on LinkedIn DMs or any shadow-DOM input. Won't find the element. Usebrowser_type(selector)
to focus, thenbrowser_click_coordinate
to type.browser_type_focused(text=...) - Clicking a "Photo" / "Attach" / "Upload" button to pick a file. This opens Chrome's NATIVE OS file picker, which is rendered outside the web page and cannot be interacted with via CDP. Your automation will hang staring at an unreachable dialog. ALWAYS use
against the underlyingbrowser_upload(selector, file_paths)
element — see the "File uploads" section above for the full pattern. This is the single most common way to wedge a browser session on compose-with-media flows (X/LinkedIn/Gmail).<input type='file'> - Keyboard shortcuts without the
field. Chrome's shortcut dispatcher ignores keyboard events that lack acode
orcode
.windowsVirtualKeyCode
populates these automatically; rawbrowser_press(..., modifiers=[...])
calls fromInput.dispatchKeyEvent
may not.browser_evaluate - Taking a screenshot more than 10s after the last interaction and expecting the highlight to still be visible. The overlay fades after 10s. Take the screenshot sooner, or re-trigger the interaction.
- Expecting
to return when you specifiedbrowser_navigate
on a busy site. networkidle is approximate — some sites keep a websocket or analytics beacon open forever. Usewait_until="networkidle"
or"load"
for reliable timing."domcontentloaded"
Dead CDP sessions and auto-recovery
If Chrome detaches the debugger for its own reasons (tab closed, user opened DevTools manually, cross-origin navigation,
chrome:// page loaded), the bridge detects the "target closed" / "not attached" error on the next call and automatically reattaches + retries once. You don't need to handle this yourself.
If reattach also fails, you'll get the underlying CDP error string — that's a real problem, usually the tab is gone.
browser_evaluate
is a last-resort escape hatch
browser_evaluateBefore using
, try these first — in this order:browser_evaluate
+browser_screenshot
— works on every site regardless of shadow DOM, iframes, obfuscated classes. This is the default path for "click a thing you can see."browser_click_coordinate
— for typing into ANY input/contenteditable, including Lexical and Draft.js. Handles click-focus-insert with built-in retries. Do not callbrowser_type(use_insert_text=True, text=...)
via evaluate; this tool already does it correctly.document.execCommand('insertText')
orbrowser_shadow_query
with thebrowser_get_rect(selector)
shadow-piercing syntax — for selector-based lookups across shadow roots.>>>
/browser_get_text
— for reading element state by selector.browser_get_attribute
— for dumping the accessibility tree of the page.browser_snapshot
If all five of those fit your goal, do not use
. Each evaluate call is a small LLM round-trip of ~30-100 tokens of JS plus a JSON response; five of them burn more context than a single screenshot-and-coordinate does, with less reliability.browser_evaluate
Anti-patterns — stop immediately if you catch yourself doing these
- Trying multiple
variants when the first returnedquerySelectorAll
. Different selectors on the same page rarely work if the first guess failed — modern SPAs obfuscate class names at build time. After one empty result, switch to[]
+browser_screenshot
. Do not writebrowser_click_coordinate
, then.artdeco-list__item
, then[data-test-incoming-invitation-card]
— you are already on the wrong path.[class*="invitation"] - Writing
recursive shadow-DOM traversal functions. Usewalk(root)
— it traverses at the CDP level (native C++), not by re-running a recursive JS function every call.browser_shadow_query - Calling
to type into a contenteditable. Usedocument.execCommand('insertText', ...)
. The high-level tool handles the exact same Lexical/Draft.js case but with click-focus-retry logic built in.browser_type(use_insert_text=True, text='...') - Accessing
. Rarely works (cross-origin, late hydration) and when it does, the code is brittle. Useiframe.contentDocument
to see the iframe, thenbrowser_screenshot
to interact.browser_click_coordinate - Using
on a Trusted Types site (LinkedIn, GitHub). The assignment is silently dropped. UseinnerHTML = "<...>"
+createElement
if you must inject DOM — but first, ask whether you really need to.appendChild - Triggering React/Vue state via synthetic
. Frameworks watch for real browser events. UsedispatchEvent
,browser_click_coordinate
, orbrowser_press
— all go through CDP's native event pipeline.browser_type
Legitimate uses (when nothing semantic fits)
- Reading a computed style,
,window.innerWidth/Height
, or other layout values the tools don't expose.document.scrollingElement.scrollTop - Firing a one-shot site-specific API call (analytics beacon, feature-flag toggle).
- Stripping
before navigating away from a page with an unsent draft (LinkedIn, Gmail).onbeforeunload - Detecting whether a specific shadow-root host exists before a follow-up screenshot.
In all of these cases the script is SHORT (< 10 lines) and the result is CONSUMED (read, then acted on), not further probed.
Login & auth walls
- If you see a "Log in" or "Sign up" prompt, report the auth wall to user immediately — do NOT attempt to log in.
- Check for cookie consent banners and dismiss them if they block content.
Error recovery
- If a tool fails, retry once with the same approach.
- If it fails a second time, STOP retrying and switch approach.
- If
fails, trybrowser_snapshot
with a specific small selector as fallback.browser_get_text - If
fails or page seems stale,browser_open
, thenbrowser_stop
, then retry.browser_start
Verified workflows
These sequences have been empirically verified against real production sites on 2026-04-11.
Search on X and read the live dropdown
browser_navigate("https://x.com/explore", wait_until="load") # Wait for SPA hydration sleep(3) browser_wait_for_selector("input[data-testid='SearchBox_Search_Input']", timeout_ms=5000) rect = browser_get_rect("input[data-testid='SearchBox_Search_Input']") browser_click_coordinate(rect.cx, rect.cy) browser_type("input[data-testid='SearchBox_Search_Input']", "openai", clear_first=True) # Screenshot now shows live search suggestions browser_screenshot() browser_press("Escape", selector="input[data-testid='SearchBox_Search_Input']")
Search Reddit (shadow DOM)
browser_navigate("https://www.reddit.com/r/programming/", wait_until="load") sleep(2) # Shadow-pierce the nested search input sq = browser_shadow_query("reddit-search-large >>> #search-input") browser_click_coordinate(sq.rect.cx, sq.rect.cy) # Typing can't use selector (shadow); use browser_type_focused on the focused input browser_type_focused(text="python") browser_screenshot() browser_press("Escape")
Search LinkedIn and dismiss without submitting
browser_navigate("https://www.linkedin.com/feed/", wait_until="load") sleep(3) browser_wait_for_selector("input[data-testid='typeahead-input']", timeout_ms=5000) rect = browser_get_rect("input[data-testid='typeahead-input']") browser_click_coordinate(rect.cx, rect.cy) browser_type("input[data-testid='typeahead-input']", "anthropic", clear_first=True) # Dropdown shows real live suggestions browser_screenshot() browser_press("Escape", selector="input[data-testid='typeahead-input']")
Debugging checklist when a click / type "didn't work"
- Send button stays disabled after typing? Two possible causes. (a) You didn't click the input first, so React never saw a native focus event.
now clicks automatically — but if you're using rawbrowser_type
, click first yourself. (b) You're using per-characterInput.dispatchKeyEvent
on a Lexical / Draft.js editor, and those editors dropped the keys because they listen forkeyDown
with a specific shape. Switch tobeforeinput
(which now usesbrowser_type(selector, text)
by default) or, at a lower level, call CDPInput.insertText
directly. AlwaysInput.insertText
the submit button'sbrowser_evaluate
/disabled
state before clicking send; if still disabled after those fixes, the framework never saw real input.aria-disabled - Did the selector match anything? Run
— if it returnsbrowser_get_rect(selector)
or zero rect, the element isn't laid out yet. Wait longer or use a different selector.visible=False - Is the element inside a shadow root? Try
. If your selector is light-DOM only, switch to the screenshot + coordinate strategy.browser_shadow_query(path) - Did the click hit something on top of the element? Register a temporary event listener via
on the target element, click, then readbrowser_evaluate
to see what actually received the click. If something else is intercepting (overlay, modal, floating button), dismiss it first.window.__hits - Did
find the element but fail to insert text? Some editors (Draft.js on X, ProseMirror on some sites, Monaco) require a smalltype_text
between keystrokes. Trydelay_ms
.delay_ms=20 - Is this a keyboard shortcut that doesn't fire? Make sure you're using
— not rawbrowser_press(key, modifiers=[...])
withbrowser_evaluate
. Chrome ignores shortcut key events that lackdispatchEvent
andcode
.windowsVirtualKeyCode - Did the navigation actually complete? Check the return value of
— it now returns a realbrowser_navigate
andurl
. An empty title usually means a blank page or a hung load.title - Is your screenshot stale? The highlight overlay stays for 10 s; if the screenshot was taken later, the annotation is gone but the click was real. Check the logs of
to see the coordinates that were actually sent.browser_click_coordinate