selenium-recorder-mcp
Records browser interactions (clicks, DOM mutations, console logs, JS errors) via Chrome DevTools Protocol while you manually navigate, and provides tools to retrieve and analyze recordings for test generation.
README
Selenium Test Recorder MCP Server
MCP server for recording comprehensive browser interactions using Chrome DevTools Protocol (CDP). Records DOM mutations, console logs, and JavaScript errors while you manually interact with a website.
Features
- Click Tracking: Captures every click with full element details (XPath, CSS selector, attributes, text content, coordinates)
- Comprehensive Recording: Captures DOM mutations, console logs, and JS errors
- Manual Interaction: Record while you manually navigate and interact with browser
- Sensitive Field Masking: Automatically masks password fields and other sensitive data
- Smart Filtering: Query recordings by event type, time range, or pagination - prevents large context dumps
- Metadata-First: See event breakdown before retrieving data - safe by default
- JSON Output: Structured data for analysis and test generation
- Claude Code Integration: Use via MCP tools in Claude Code
What's New
Smart Query Filtering (Breaking Change)
The get_recording tool now returns metadata only by default to prevent large responses that can fill up Claude's context window.
Before:
get_recording(session_id) → Returns all events (could be 70k+ tokens)
Now:
get_recording(session_id) → Returns metadata only (event breakdown, file path)
get_recording(session_id, limit=50) → Returns first 50 events
get_recording(session_id, event_types=["click"]) → Returns only click events
Benefits:
- ✅ No more accidental context overflow warnings
- ✅ See what's available before requesting data
- ✅ Filter to exactly what you need
- ✅ Paginate large recordings
Migration: Add any filter parameter to get events (e.g., limit, event_types, offset)
See Querying Recordings with Filters for details.
Installation
cd selenium-recorder-mcp
uv venv
source .venv/bin/activate
uv pip install -e .
Requirements
- Python 3.10+
- Google Chrome (or Chromium-based browser)
- macOS, Linux, or Windows
Quick Start (Standalone Script)
Easiest way to use - no MCP setup required:
cd selenium-recorder-mcp
source .venv/bin/activate
python record.py https://www.example.com
This will:
- Open Chrome at the URL
- Start recording clicks, DOM mutations, console logs, and JS errors
- Let you manually interact (login, navigate, click, etc.)
- Press ENTER when done to stop recording
- Save recording to
recordings/folder as JSON
Example output:
✅ Recording started!
📝 Session ID: 550e8400-e29b-41d4-a716-446655440000
🌐 Chrome opened at: https://www.example.com
👉 Interact with the browser now...
Press ENTER when done to stop recording
[... you interact with the browser ...]
⏹️ Stopping recording...
✅ Recording saved to: recordings/550e8400_20250930_143022.json
📊 Total events: 247
📈 Summary:
- Clicks: 12
- DOM mutations: 189
- Console logs: 52
- JS errors: 6
- Masked events: 3
Setup (MCP Server Integration)
Configure Claude Code MCP
Method 1: Using claude mcp add command (Recommended)
From your terminal, run:
claude mcp add selenium-recorder \
--scope user \
-- /bin/zsh -lc '
cd /path/to/selenium-recorder-mcp &&
exec ./.venv/bin/python -m src.server
'
Replace /path/to/selenium-recorder-mcp with your actual installation directory.
If Chrome is not in the default location, add the --env flag:
claude mcp add selenium-recorder \
--scope user \
--env CHROME_PATH="/Applications/Google Chrome.app/Contents/MacOS/Google Chrome" \
-- /bin/zsh -lc '
cd /path/to/selenium-recorder-mcp &&
exec ./.venv/bin/python -m src.server
'
Default Chrome paths:
- macOS:
/Applications/Google Chrome.app/Contents/MacOS/Google Chrome - Linux:
/usr/bin/google-chrome - Windows:
C:\Program Files\Google\Chrome\Application\chrome.exe
For bash users, replace /bin/zsh -lc with /bin/bash -lc
Notes:
--scope usermakes it available across all your projects- The
-lcflag ensures a login shell with proper environment setup execreplaces the shell process with Python for cleaner process management
Method 2: Manual JSON configuration (Alternative)
Add to your Claude Code MCP settings (~/.claude/mcp_settings.json):
{
"mcpServers": {
"selenium-recorder": {
"command": "/path/to/selenium-recorder-mcp/.venv/bin/python",
"args": ["-m", "src.server"],
"cwd": "/path/to/selenium-recorder-mcp",
"env": {
"CHROME_PATH": "/Applications/Google Chrome.app/Contents/MacOS/Google Chrome"
}
}
}
}
Important: Use absolute paths for both command and cwd fields.
Usage
Method 1: Standalone Script (Recommended for getting started)
python record.py https://www.sport5.co.il
- Chrome opens automatically
- Manually interact (login, navigate, click)
- Press ENTER to stop
- Recording saved to
recordings/folder
Then share the recording JSON file with Claude Code for analysis or test generation!
Method 2: MCP Server (Advanced - Claude Code Integration)
After configuring MCP settings and restarting Claude Code:
Workflow:
- Start Recording - Opens Chrome and begins recording
- Manual Interaction - Login, navigate, interact with the site
- Stop Recording - Saves events to JSON file
- Analyze/Generate - Use Claude Code to analyze or generate test code
MCP Tools:
start_recording
Start recording browser session.
Parameters:
url(optional): URL to navigate to on startsensitive_selectors(optional): Additional CSS selectors for sensitive fields
Returns: Session ID
Example:
start_recording with url "https://example.com"
stop_recording
Stop recording and save to JSON.
Parameters:
session_id(required): Session ID from start_recording
Returns: File path to saved recording
Example:
stop_recording with session_id "abc-123-def"
get_recording
Retrieve recording metadata or filtered events.
Parameters:
session_id(required): Session ID to retrieveevent_types(optional): Array of event types to filter (e.g.,["click", "console_log"])limit(optional): Maximum number of events to returnoffset(optional): Number of events to skip (for pagination)from_timestamp(optional): ISO timestamp - only return events after this timeto_timestamp(optional): ISO timestamp - only return events before this time
Returns:
- Without filters: Metadata only (event type breakdown, file path, total count) - prevents large responses
- With any filter: Filtered events matching criteria
Examples:
# Get metadata only (safe, no large dumps)
get_recording with session_id "abc-123"
# Get first 50 events
get_recording with session_id "abc-123" and limit 50
# Get only click events
get_recording with session_id "abc-123" and event_types ["click"]
# Get clicks and DOM changes
get_recording with session_id "abc-123" and event_types ["click", "dom_attribute_modified"]
analyze_recording
Get summary statistics.
Parameters:
session_id(required): Session ID to analyze
Returns: Event summary and counts
Example Session with Claude Code
You: Start recording https://myapp.com/login
Claude: [calls start_recording]
Recording started. Session ID: 550e8400-e29b-41d4-a716-446655440000
Chrome browser opened at https://myapp.com/login.
Interact with the browser manually. Call stop_recording when done.
[You manually login and navigate through the app]
You: Stop recording with session_id 550e8400-e29b-41d4-a716-446655440000
Claude: [calls stop_recording]
Recording stopped and saved to: recordings/550e8400-e29b-41d4-a716-446655440000_20250115_143022.json
Total events recorded: 247
You: What's in the recording?
Claude: [calls get_recording with session_id only - no filters]
Session: 550e8400-e29b-41d4-a716-446655440000
File: recordings/550e8400-e29b-41d4-a716-446655440000_20250115_143022.json
Total events: 247
Event type breakdown:
dom_attribute_modified: 89
console_log: 52
dom_set_child_nodes: 45
click: 12
dom_character_data_modified: 43
js_error: 6
ℹ️ Use filters to retrieve events (limit, event_types, offset, timestamps)
Or read the file directly.
You: Show me just the click events
Claude: [calls get_recording with event_types=["click"]]
Session: 550e8400-e29b-41d4-a716-446655440000
Events: 12/247
Filters: {"event_types": ["click"]}
[Returns 12 click events with full details]
You: Generate a Selenium test from the click events
Claude: [uses the 12 click events to generate test code]
Querying Recordings with Filters
The get_recording tool now supports powerful filtering to prevent large context dumps and retrieve only the data you need.
Default Behavior (Metadata Only)
Without any filters, get_recording returns metadata only:
get_recording with session_id "abc-123"
Returns:
- Session information (URL, timestamps, file path)
- Total event count
- Event type breakdown (shows count per event type)
- Usage instructions
- No events - prevents accidentally filling context with large responses
This is safe by default and helps you understand what's in the recording before requesting specific data.
Retrieving Events with Filters
To get actual events, use any filter parameter:
Limit Events (Pagination)
# First 50 events
get_recording with session_id "abc-123" and limit 50
# Next 50 events (pagination)
get_recording with session_id "abc-123" and limit 50 and offset 50
Filter by Event Type
# Only clicks
get_recording with session_id "abc-123" and event_types ["click"]
# Clicks and DOM mutations
get_recording with session_id "abc-123" and event_types ["click", "dom_attribute_modified", "dom_set_child_nodes"]
# Exclude noisy console logs - get everything else
get_recording with session_id "abc-123" and event_types ["click", "js_error", "dom_attribute_modified"]
Time Range Filtering
# Events after a specific time
get_recording with session_id "abc-123" and from_timestamp "2025-01-15T14:30:00"
# Events in a time window
get_recording with session_id "abc-123" and from_timestamp "2025-01-15T14:30:00" and to_timestamp "2025-01-15T14:35:00"
Combined Filters
# Click events only, first 100
get_recording with session_id "abc-123" and event_types ["click"] and limit 100
# Recent errors only
get_recording with session_id "abc-123" and event_types ["js_error"] and from_timestamp "2025-01-15T14:30:00"
Filter Extensibility
Filters are event-type agnostic and work with any event type:
- Current event types:
click,console_log,js_error,dom_attribute_modified,dom_set_child_nodes,dom_character_data_modified,document_updated - When new event types are added (e.g.,
window_resize,network_request), they're immediately filterable with no code changes - Just use the event type name in the
event_typesarray
Typical Workflow
- Get overview - Call without filters to see event breakdown
- Filter what you need - Request specific event types or ranges
- Paginate if needed - Use
limitandoffsetfor large result sets - Generate/analyze - Use filtered data for test generation or analysis
Example:
# Step 1: What's in the recording?
get_recording with session_id "abc-123"
→ Shows: 514 events (click: 45, console_log: 320, js_error: 12, ...)
# Step 2: I only care about user interactions
get_recording with session_id "abc-123" and event_types ["click"]
→ Returns: 45 click events
# Step 3: Generate test from clicks
[Claude uses 45 click events to generate Selenium test]
Output Format
Recordings are saved as JSON:
{
"session_id": "550e8400-e29b-41d4-a716-446655440000",
"url": "https://example.com",
"start_time": "2025-01-15T14:30:15.123456",
"end_time": "2025-01-15T14:35:22.654321",
"events": [
{
"type": "dom_attribute_modified",
"timestamp": "2025-01-15T14:30:18.123456",
"data": {
"node_id": 42,
"name": "value",
"value": "***MASKED***",
"_masked": true
}
},
{
"type": "console_log",
"timestamp": "2025-01-15T14:30:20.123456",
"data": {
"level": "info",
"args": ["User logged in successfully"]
}
},
{
"type": "js_error",
"timestamp": "2025-01-15T14:30:25.123456",
"data": {
"message": "TypeError: Cannot read property 'id' of undefined",
"stack": "..."
}
}
],
"metadata": {
"saved_at": "2025-01-15T14:35:22.987654",
"event_count": 247
}
}
Sensitive Field Masking
By default, these fields are masked:
input[type="password"]input[name*="password"]input[name*="passwd"]input[id*="password"]input[name*="secret"]input[name*="token"]
Add custom sensitive selectors:
start_recording with sensitive_selectors ["input[name='credit-card']", "input[id='ssn']"]
Event Types
Click Events
click: User click interactions with comprehensive element details:- Identification: Tag name, ID, classes, all attributes
- Content: Text content (first 200 chars), innerHTML (first 500 chars)
- Selectors: XPath and CSS selector for precise element location
- Links/Media: href (for links), src (for images)
- Position: Click coordinates (x, y) and page position
- Context: Viewport size and current URL
Example click event:
{
"type": "click",
"timestamp": "2025-01-15T14:30:22.123456",
"data": {
"tagName": "A",
"id": "login-button",
"className": "btn btn-primary",
"classList": ["btn", "btn-primary"],
"attributes": {"href": "/login", "target": "_self"},
"textContent": "Login",
"innerHTML": "Login",
"xpath": "//*[@id=\"login-button\"]",
"cssSelector": "#login-button",
"href": "https://example.com/login",
"coordinates": {"x": 150, "y": 200, "pageX": 150, "pageY": 200},
"viewport": {"width": 1920, "height": 1080},
"url": "https://example.com"
}
}
DOM Events
document_updated: Document structure changeddom_set_child_nodes: Child nodes added/modifieddom_attribute_modified: Element attribute changeddom_character_data_modified: Text content changed
Runtime Events
console_log: Console output (log, info, warn, error)js_error: JavaScript exception thrown
Troubleshooting
Chrome doesn't open
- Check
CHROME_PATHis correct - Ensure Chrome is installed
- Try absolute path to Chrome executable
No events recorded
- Check Chrome DevTools console for errors
- Ensure page allows remote debugging
- Some CSP policies may block CDP
MCP server fails to connect from another folder
- Use absolute paths in
mcp_settings.jsonfor thecommandfield - Example:
"/Users/username/selenium-recorder-mcp/.venv/bin/python" - The
cwdsetting only affects where the process runs, not where the command is found
Sensitive data not masked
- Add custom selectors via
sensitive_selectors - Check field naming patterns match defaults
- Review event_processor.py patterns
Use Cases
- Test Generation: Record manual workflow, generate Selenium tests
- Bug Reproduction: Capture exact DOM state during error
- Performance Analysis: Track DOM mutations and console logs
- Debugging: See all browser events during interaction
Architecture
┌─────────────────┐
│ Claude Code │
│ (MCP Client) │
└────────┬────────┘
│
│ MCP Protocol
│
┌────────▼────────┐
│ server.py │ MCP Tools
│ (MCP Server) │ - start_recording
└────────┬────────┘ - stop_recording
│ - get_recording
│ - analyze_recording
│
┌────────▼──────────────────────────┐
│ cdp_recorder.py │
│ - Launch Chrome w/ debug port │
│ - Connect via CDP │
│ - Inject click tracking JS │
│ - Listen to DOM/Runtime events │
└────────┬──────────────────────────┘
│
┌────────▼──────────────────────────┐
│ event_processor.py │
│ - Mask sensitive fields │
│ - Enrich events │
│ - Analyze patterns │
└────────┬──────────────────────────┘
│
┌────────▼──────────────────────────┐
│ storage.py │
│ - Save as JSON │
│ - Load recordings │
│ - List/delete │
└───────────────────────────────────┘
Development
Run server directly:
python -m src.server
License
MIT
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.