Skip to main content

Browser Provider

The Browser Provider enables automated web browser interactions for testing complex web applications and JavaScript-heavy websites where simpler providers are not sufficient.

This provider uses Playwright to control headless browsers, allowing you to navigate pages, interact with elements, and extract data from dynamic websites. Playwright supports Chromium (Chrome, Edge), Firefox, and WebKit (Safari engine) browsers.

When to Use the Browser Provider

The Browser Provider should only be used when simpler alternatives are not possible:

  1. Try these first:

  2. Use Browser Provider only when:

    • The application requires JavaScript execution to render content
    • You need to interact with complex UI elements (dropdowns, modals, etc.)
    • Authentication requires browser-based workflows (OAuth, SSO)
    • You need to test actual user interactions (clicks, typing, scrolling)

Important Considerations

When using browser automation:

  1. Rate Limiting: Always implement delays between requests to avoid overwhelming servers
  2. Anti-Bot Detection: Many websites employ anti-bot measures that can detect and block automated browsers
  3. Resource Usage: Browser automation is 10-100x slower than direct API calls and consumes significant CPU/memory
  4. Legal Compliance: Always check the website's Terms of Service and robots.txt before automating

Prerequisites

Playwright is a peer dependency of promptfoo, so you will need to install it separately:

npm install playwright @playwright/browser-chromium playwright-extra puppeteer-extra-plugin-stealth

Note: Currently, promptfoo's browser provider only supports Chromium-based browsers (Chrome, Edge). The provider uses playwright-extra with the Chromium engine for enhanced stealth capabilities.

Configuration

To use the Browser Provider, set the provider id to browser and define a series of steps to execute:

providers:
- id: browser
config:
steps:
- action: navigate
args:
url: 'https://example.com'
- action: type
args:
selector: '#search-input'
text: '{{prompt}}'
- action: click
args:
selector: '#search-button'
- action: extract
args:
selector: '#results'
name: searchResults
transformResponse: 'extracted.searchResults'

Connecting to Existing Browser Sessions

You can connect to an existing Chrome browser session (e.g., with OAuth authentication already completed):

providers:
- id: browser
config:
connectOptions:
debuggingPort: 9222 # Chrome debugging port

steps:
# Your test steps here

Setup Instructions:

  1. Start Chrome with debugging: chrome --remote-debugging-port=9222 --user-data-dir=/tmp/test
  2. Complete authentication manually
  3. Run your tests

Connection Options:

  • debuggingPort: Port number for Chrome DevTools Protocol (default: 9222)
  • mode: Connection mode - 'cdp' (default) or 'websocket'
  • wsEndpoint: Direct WebSocket endpoint (when using mode: 'websocket')

Supported Actions

The Browser Provider supports the following actions:

Core Actions

1. navigate - Load a webpage

Navigate to a specified URL.

- action: navigate
args:
url: 'https://example.com/search?q={{query}}'

2. click - Click an element

Click on any clickable element (button, link, etc.).

- action: click
args:
selector: 'button[type="submit"]'
optional: true # Won't fail if element doesn't exist

3. type - Enter text

Type text into input fields, textareas, or any editable element.

- action: type
args:
selector: 'input[name="username"]'
text: '{{username}}'

Special keys:

  • <enter> - Press Enter key
  • <tab> - Press Tab key
  • <escape> - Press Escape key

4. extract - Get text content

Extract text from any element. The extracted content is available in transformResponse.

- action: extract
args:
selector: '.result-title'
name: title # Access as extracted.title

5. wait - Pause execution

Wait for a specified duration (in milliseconds).

- action: wait
args:
ms: 3000 # Wait 3 seconds

6. waitForNewChildren - Wait for dynamic content

Wait for new elements to appear under a parent element. Useful for content loaded via AJAX.

- action: waitForNewChildren
args:
parentSelector: '#results-container'
delay: 500 # Check every 500ms
timeout: 10000 # Max wait time 10 seconds

7. screenshot - Capture the page

Take a screenshot of the current page state.

- action: screenshot
args:
path: 'screenshot.png'
fullPage: true # Capture entire page, not just viewport

Action Parameters

ActionRequired ArgsOptional ArgsDescription
navigateurl-URL to navigate to
clickselectoroptionalCSS selector of element to click
typeselector, text-CSS selector and text to type
extractselector, name-CSS selector and variable name
waitms-Milliseconds to wait
waitForNewChildrenparentSelectordelay, timeoutParent element to watch
screenshotpathfullPageFile path to save screenshot

Response Parsing

Use the transformResponse config option to extract specific data from the results. The parser receives an object with two properties:

  • extracted: An object containing named results from extract actions
  • finalHtml: The final HTML content of the page after all actions are completed

Variables and Templating

You can use Nunjucks templating in your configuration, including the {{prompt}} variable and any other variables passed in the test context.

providers:
- id: browser
config:
steps:
- action: navigate
args:
url: 'https://example.com/search?q={{prompt}}'
- action: extract
args:
selector: '#first-result'
name: topResult
transformResponse: 'extracted.topResult'

tests:
- vars:
prompt: 'What is the capital of France?'

Using as a Library

If you are using promptfoo as a node library, you can provide the equivalent provider config:

{
// ...
providers: [{
id: 'browser',
config: {
steps: [
{ action: 'navigate', args: { url: 'https://example.com' } },
{ action: 'type', args: { selector: '#search', text: '{{prompt}}' } },
{ action: 'click', args: { selector: '#submit' } },
{ action: 'extract', args: { selector: '#results' }, name: 'searchResults' }
],
transformResponse: (extracted, finalHtml) => extracted.searchResults,
}
}],
}

Reference

Supported config options:

OptionTypeDescription
headlessbooleanWhether to run the browser in headless mode. Defaults to true.
cookiesstring | { name: string; value: string; domain?: string; path?: string; }[]A string or array of cookies to set on the browser
transformResponsestring | FunctionA function or string representation of a function to parse the response. Receives an object with extracted and finalHtml parameters and should return a ProviderResponse
stepsBrowserAction[]An array of actions to perform in the browser
timeoutMsnumberThe maximum time in milliseconds to wait for the browser operations to complete

Note: All string values in the config support Nunjucks templating. This means you can use the {{prompt}} variable or any other variables passed in the test context.

Browser Support

While Playwright supports multiple browsers (Chromium, Firefox, and WebKit), promptfoo's browser provider currently only implements Chromium support. This includes:

  • Chrome - Google's browser
  • Edge - Microsoft's Chromium-based browser
  • Chromium - Open-source browser project

The implementation uses playwright-extra with the Chromium engine for enhanced stealth capabilities to avoid detection.

Supported Browser Actions

The steps array in the configuration can include the following actions:

ActionDescriptionRequired ArgsOptional Args
navigateNavigate to a specified URLurl: string
clickClick on an elementselector: stringoptional: boolean
extractExtract text content from an elementselector: string, name: string
screenshotTake a screenshot of the pagepath: stringfullPage: boolean
typeType text into an input fieldselector: string, text: string
waitWait for a specified amount of timems: number
waitForNewChildrenWait for new child elements to appear under a parentparentSelector: stringdelay: number, timeout: number

Each action in the steps array should be an object with the following structure:

{
action: string;
args: {
[key: string]: any;
};
name?: string;
}

Each step in the steps array should have the following structure:

  • action: Specifies the type of action to perform (e.g., 'navigate', 'click', 'type').
  • args: Contains the required and optional arguments for the action.
  • name (optional): Used to name extracted content in the 'extract' action.

Steps are executed sequentially, enabling complex web interactions.

All string values in args support Nunjucks templating, allowing use of variables like {{prompt}}.

Advanced Features

Playwright Recorder Tools

The easiest way to create browser automation scripts is to record your interactions:

The Playwright Recorder Chrome Extension is particularly helpful for quickly generating selectors:

  1. Install the extension from the Chrome Web Store
  2. Navigate to your target website
  3. Click the extension icon and start recording
  4. Perform your actions (click, type, etc.)
  5. Stop recording and copy the generated selectors/code
  6. Adapt the code for promptfoo's browser provider format

This extension is especially useful because it:

  • Shows selectors in real-time as you hover over elements
  • Generates multiple selector options (CSS, text, XPath)
  • Allows you to copy individual selectors without recording full actions

Playwright Inspector (All Browsers)

For cross-browser recording, use Playwright's built-in recorder:

npx playwright codegen https://example.com

This opens an interactive browser window where you can perform actions and see generated code in real-time. You can choose between Chromium, Firefox, or WebKit.

Selector Strategies

Playwright supports various selector strategies:

StrategyExampleDescription
CSS#submit-buttonStandard CSS selectors
Texttext=SubmitFind elements by text content
Rolerole=button[name="Submit"]ARIA role-based selectors
Test IDdata-testid=submitData attribute selectors
XPathxpath=//button[@type="submit"]XPath expressions

For the most reliable selectors:

  • Prefer stable attributes like IDs and data-testid
  • Use role-based selectors for accessibility
  • Avoid position-based selectors that can break with layout changes

Debugging

1. Disable Headless Mode

See exactly what's happening in the browser:

providers:
- id: browser
config:
headless: false # Opens visible browser window

2. Enable Debug Logging

Get detailed information about each action:

npx promptfoo@latest eval --verbose

3. Take Screenshots

Capture the page state during execution:

steps:
- action: navigate
args:
url: 'https://example.com'
- action: screenshot
args:
path: 'debug-{{_attempt}}.png'

Performance Optimization

  1. Use headless mode in production: It's faster and uses fewer resources
  2. Minimize wait times: Only wait as long as necessary
  3. Batch operations: Group related actions together
  4. Reuse browser contexts: For multiple tests against the same site

Best Practices for Rate Limiting

Implementing proper rate limiting is crucial to avoid detection and server overload:

providers:
- id: browser
config:
steps:
# Always start with a respectful delay
- action: wait
args:
ms: 2000

- action: navigate
args:
url: 'https://example.com'

# Wait between actions
- action: wait
args:
ms: 1000

- action: click
args:
selector: '#button'

# Final delay before next request
- action: wait
args:
ms: 3000

Tips for avoiding detection:

  • Randomize delays between actions (1-3 seconds)
  • Use the stealth plugin (included with playwright-extra)
  • Avoid patterns that look automated
  • Consider using different user agents
  • Respect robots.txt and rate limits

Dealing with Anti-Bot Measures

Many websites implement anti-bot detection systems (like Cloudflare, reCAPTCHA, etc.). Here's how to handle common scenarios:

Common Anti-Bot Challenges

ChallengeDetection MethodMitigation Strategy
Browser fingerprintingJavaScript checks for automationStealth plugin helps mask automation
Behavioral analysisMouse movements, typing patternsAdd realistic delays and interactions
IP rate limitingToo many requests from one IPImplement proper delays, use proxies cautiously
CAPTCHA challengesHuman verification testsConsider if the site allows automation
User-Agent detectionChecking for headless browsersUse realistic user agent strings

Example with Anti-Bot Considerations

providers:
- id: browser
config:
headless: false # Some sites detect headless mode
steps:
# Human-like delay before starting
- action: wait
args:
ms: 3000

- action: navigate
args:
url: '{{url}}'

# Wait for any anti-bot checks to complete
- action: wait
args:
ms: 5000

# Type slowly like a human would
- action: type
args:
selector: '#search'
text: '{{query}}'
delay: 100 # Delay between keystrokes

Note: If a website has strong anti-bot measures, it's often a sign that automation is not welcome. Always respect the website owner's wishes and consider reaching out for API access instead.

Example: Testing a Login Flow

Here's a complete example testing a login workflow:

# yaml-language-server: $schema=https://promptfoo.dev/config-schema.json
description: Test login functionality

prompts:
- 'Login with username {{username}} and password {{password}}'

providers:
- id: browser
config:
headless: true
steps:
- action: navigate
args:
url: 'https://example.com/login'

- action: type
args:
selector: '#username'
text: '{{username}}'

- action: type
args:
selector: '#password'
text: '{{password}}'

- action: click
args:
selector: 'button[type="submit"]'

- action: wait
args:
ms: 2000

- action: extract
args:
selector: '.welcome-message'
name: welcomeText

transformResponse: |
return {
output: extracted.welcomeText,
success: extracted.welcomeText.includes('Welcome')
};

tests:
- vars:
username: 'testuser'
password: 'testpass123'
assert:
- type: javascript
value: output.success === true

Troubleshooting

Common Issues and Solutions

IssueCauseSolution
"Element not found"Selector incorrect or element not loaded• Verify selector in DevTools
• Add wait before action
• Check if element is in iframe
"Timeout waiting for selector"Page loads slowly or element never appears• Increase timeout
• Add explicit wait actions
• Check for failed network requests
"Access denied" or 403 errorsAnti-bot detection triggered• Use headless: false
• Add more delays
• Check if automation is allowed
"Click intercepted"Element covered by overlay• Wait for overlays to disappear
• Scroll element into view
• Use force click option
Inconsistent resultsTiming or detection issues• Add consistent delays
• Use stealth plugin
• Test during off-peak hours

Debugging Anti-Bot Detection

If you suspect anti-bot measures are blocking your automation:

providers:
- id: browser
config:
headless: false # Always start with headed mode for debugging
steps:
- action: navigate
args:
url: '{{url}}'

- action: screenshot
args:
path: 'debug-landing.png' # Check if you hit a challenge page

- action: wait
args:
ms: 10000 # Longer wait to see what happens

- action: screenshot
args:
path: 'debug-after-wait.png'

Useful Resources


For more examples, check out the headless-browser example in our GitHub repository.