Chrome Browser Control
Provides tools to control the Google Chrome browser, including listing tabs, navigating, clicking, typing, scrolling, and taking DOM snapshots.
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@Chrome Browser Controlnavigate to google.com"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
Chrome Browser Control
Local Chrome-profile control for stdio MCP hosts.
This project exposes browser-control MCP tools through a Manifest V3 Chrome extension connected to a loopback WebSocket broker. Configure your MCP host to launch the stdio adapter with the same pairing token you enter in the extension.
Repository: https://github.com/vkongv/chrome-browser-control
Prerequisites
Node.js 18+
Google Chrome
Related MCP server: superpowers-chrome
Install and Setup
Clone the repository:
git clone https://github.com/vkongv/chrome-browser-control.git cd chrome-browser-controlInstall dependencies and run setup:
npm install npm run setupnpm run setupwrites.env.localwithCHROME_BROWSER_CONTROL_TOKENandCHROME_BROWSER_CONTROL_PORT, prints the extension folder path, and shows copy-paste MCP config snippets. That file is gitignored — do not commit it.Continue with Load The Extension, then add MCP config and verify the connection.
To generate a pairing token manually instead of npm run setup:
node -e "console.log(crypto.randomBytes(32).toString('base64url'))"Environment Variables
CHROME_BROWSER_CONTROL_TOKEN— Required. High-entropy pairing token shared by the broker, MCP adapter, and extension popup.CHROME_BROWSER_CONTROL_PORT— WebSocket broker port (default8765).CHROME_BROWSER_CONTROL_HOST— Loopback host for the broker (default127.0.0.1).CHROME_BROWSER_CONTROL_EXTENSION_ID— Optional. Pins the broker to one installed extension ID.CHROME_BROWSER_CONTROL_DISABLE_LOCAL_ENV— Optional. Set to1to skip loading.env.local.
For manual operation outside an MCP host:
CHROME_BROWSER_CONTROL_TOKEN='<generated-token>' npm run broker
CHROME_BROWSER_CONTROL_TOKEN='<generated-token>' npm run mcpnpm start is an alias for npm run mcp.
Load The Extension
Open Chrome with the profile you want the MCP tools to control.
Go to
chrome://extensions.Enable Developer mode.
Click "Load unpacked".
Select
/path/to/chrome-browser-control/extension.Open the Chrome Browser Control extension popup.
Keep the bridge URL at
ws://127.0.0.1:8765unless you changed the local port.Paste the generated pairing token.
Add allowed origins such as
https://example.com,http://localhost:3000, or*for all normalhttp://andhttps://pages.Click "Save and reconnect".
The extension may ask for host permission for the allowed origins. Denying that request prevents page actions for those origins.
Using * is convenient for local development, but it exposes every normal web page in the current Chrome profile to MCP tools. Prefer explicit origins when you only need a few sites.
MCP Host Configuration
Paste a snippet from npm run setup into Cursor, Claude Desktop, Codex, or another stdio MCP host. To print host-specific config again later:
npm run --silent mcp-config -- --host cursor
npm run --silent mcp-config -- --host claude
npm run --silent mcp-config -- --host codex
npm run --silent mcp-config -- --host yamlUse absolute paths because many MCP hosts do not apply a per-server working directory. The exact key names vary by host, but Claude Desktop, Codex, Cursor, and similar MCP hosts generally need a command, args, and env block for a stdio MCP server. This project should work with any stdio MCP host; verify host-specific config syntax in that host's documentation.
YAML-style example:
mcp_servers:
chrome_browser_control:
command: "/path/to/chrome-browser-control/node_modules/.bin/tsx"
args: ["/path/to/chrome-browser-control/server/index.ts"]
env:
CHROME_BROWSER_CONTROL_TOKEN: "<generated-token>"
CHROME_BROWSER_CONTROL_PORT: "8765"
timeout: 60
connect_timeout: 30JSON-style example:
{
"mcpServers": {
"chrome_browser_control": {
"command": "/path/to/chrome-browser-control/node_modules/.bin/tsx",
"args": ["/path/to/chrome-browser-control/server/index.ts"],
"env": {
"CHROME_BROWSER_CONTROL_TOKEN": "<generated-token>",
"CHROME_BROWSER_CONTROL_PORT": "8765"
}
}
}
}If your MCP host uses a config file, keep it private and outside the repository.
Verify
Run the setup checker:
npm run doctorThen confirm from your MCP host by calling the browser_status tool. When ready, extension.status and ping.status should reflect a live bridge connection, and extension.allowedOrigins should show your configured scope.
Tools
browser_status: checks whether the MCP adapter can reach the broker and whether the Chrome extension answersping. When ready,extension.statusandping.statusreflect the live bridge connection (not a stale disconnected default),extension.allowedOriginsshows the configured scope (including* (all http/https web origins)when wildcard mode is enabled), andprotocolVersion/featuresconfirm the loaded unpacked extension code.list_tabs: lists tabs whose URL origin is allowed in the extension popup. When every open tab is filtered out, returns{ tabs: [], detail, hiddenTabCount, allowedOrigins? }instead of a bare[]. Wildcard mode is labeled clearly inallowedOrigins.snapshot: returns a simplified DOM snapshot for an allowed tab. By default this is a compact automation snapshot that includes concise actionable elements, a text preview (500 chars), omitted counts, and region summaries. Passmode: "full"for verbose element metadata and atextfield (4000 chars by default). PasstextLimit(up to100000) when you need more page body text — checktextBytesOmittedto see if content was truncated.navigate: navigates the active tab or a specifiedtabIdto an allowed URL, then waits for the tab to finish loading when possible. If loading times out, the result includespending: trueand awarning.click: clicks an element by snapshot ref on an allowed tab.type: types into an element by snapshot ref on an allowed tab. Password-like fields are blocked unlessforce=true.scroll: scrolls an allowed tab bydeltaXanddeltaY.
Snapshot Modes And Refs
Default compact snapshots are designed to reduce model-context usage while preserving browser automation. A compact snapshot looks like:
{
"title": "Example Domain",
"url": "https://example.com/",
"mode": "compact",
"elements": [{ "ref": "h1", "role": "link", "label": "Learn more" }],
"omittedElements": 0,
"textPreview": "Example Domain ...",
"textBytesOmitted": 0,
"regions": []
}Use full mode only when you need the legacy verbose element metadata:
{ "mode": "full", "tabId": 123 }To read long page content (for example API docs), raise textLimit instead of using broker scripts or CDP workarounds:
{ "mode": "full", "textLimit": 100000, "tabId": 123 }Compact mode honors textLimit too; body text is returned in textPreview. When textBytesOmitted is greater than zero, increase textLimit or scroll the page and snapshot again for below-the-fold content.
Refs are per-document in-memory IDs (h...) assigned from element identity, not output order. They remain stable across DOM insertion/reorder in the same document, and click / type resolve through the content script's ref store. Navigating to a different page loads a new document, so old refs are expected to fail cleanly; take a fresh snapshot after navigation or major page changes. The ref store prunes disconnected, expired, and over-cap entries, and removes stale data-cbc-ref attributes so pruned refs cannot be reused accidentally.
Development Checks
npm test
npm run build
npm run doctor
npm run benchmark:compact-snapshots
npm auditnpm run benchmark:snapshots is an alias for the same compact-vs-full benchmark. The benchmark prints compact bytes, full bytes, and reduction percentage; compact mode should stay at least 50% smaller on the dense fixture.
After editing files under extension/, reload the unpacked extension on chrome://extensions before running browser e2e checks. A stale loaded background service worker can keep serving older behavior; browser_status should show the current protocolVersion and features marker when Chrome has loaded the latest extension code.
Limitations
This is a prototype with a shared local token, not multi-user authentication.
Browser tool calls are serialized globally at the broker.
Content scripts use DOM snapshots, not the full Chrome accessibility tree.
Refs are document-scoped in-memory handles. Run
snapshotagain after navigation, reloads, major DOM changes, or stale-ref errors.Browser history, bookmark, download, and cookie tools are intentionally not exposed.
server/cdp.tsremains only as an unused development reference and is not wired into the MCP adapter.
Security
No default token is accepted. Set
CHROME_BROWSER_CONTROL_TOKENto a high-entropy URL-safe value for both the broker and MCP adapter, then paste the same value into the extension popup.The broker binds only to loopback hosts:
127.0.0.1,localhost, or::1.The extension only connects to
ws://127.0.0.1,ws://localhost, orws://[::1]with an optional port.Page access is limited by allowed origins configured in the popup. Use explicit entries such as
https://example.com, or enter*to allow all normalhttp://andhttps://web pages. Tabs and page actions outside the configured scope are blocked.Optional
CHROME_BROWSER_CONTROL_EXTENSION_IDpins the broker to one installed extension ID.CDP fallback is not supported by the MCP adapter because it bypasses extension pairing.
Never bind the broker to a non-loopback interface or commit tokens, local config files, logs, or personal setup notes.
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/vKongv/chrome-browser-control'
If you have feedback or need assistance with the MCP directory API, please join our Discord server