by nottelabs
The web agent framework built for **speed**, **cost-efficiency**, **scale**, and **reliability**
→ Read more at: open-operator-evals • X • LinkedIn • Landing • Console

Notte provides all the essential tools for building and deploying AI agents that interact seamlessly with the web. Our full-stack framework combines AI agents with traditional scripting for maximum efficiency - letting you script deterministic parts and use AI only when needed, cutting costs by 50%+ while improving reliability. We allow you to develop, deploy, and scale your own agents and web automations, all with a single API. Read more in our documentation here 🔥
Opensource Core:
API service (Recommended)
pip install notte
patchright install --with-deps chromium
Use the following script to spinup an agent using opensource features (you'll need your own LLM API keys):
import notte from dotenv import load_dotenv load_dotenv() with notte.Session(headless=False) as session: agent = notte.Agent(session=session, reasoning_model='gemini/gemini-2.5-flash', max_steps=30) response = agent.run(task="doom scroll cat memes on google images")
We also provide an effortless API that hosts the browser sessions for you - and provide plenty of premium features. To run the agent you'll need to first sign up on the Notte Console and create a free Notte API key 🔑
from notte_sdk import NotteClient import os client = NotteClient(api_key=os.getenv("NOTTE_API_KEY")) with client.Session(headless=False) as session: agent = client.Agent(session=session, reasoning_model='gemini/gemini-2.5-flash', max_steps=30) response = agent.run(task="doom scroll cat memes on google images")
Our setup allows you to experiment locally, then drop-in replace the import and prefix notte
objects with cli
to switch to SDK and get hosted browser sessions plus access to premium features!
Rank | Provider | Agent Self-Report | LLM Evaluation | Time per Task | Task Reliability |
---|---|---|---|---|---|
🏆 | Notte | 86.2% | 79.0% | 47s | 96.6% |
2️⃣ | Browser-Use | 77.3% | 60.2% | 113s | 83.3% |
3️⃣ | Convergence | 38.4% | 31.4% | 83s | 50% |
Read the full story here: https://github.com/nottelabs/open-operator-evals
Structured output is a feature of the agent's run function that allows you to specify a Pydantic model as the response_format
parameter. The agent will return data in the specified structure.
from notte_sdk import NotteClient from pydantic import BaseModel from typing import List class HackerNewsPost(BaseModel): title: str url: str points: int author: str comments_count: int class TopPosts(BaseModel): posts: List[HackerNewsPost] client = NotteClient() with client.Session(headless=False, browser_type="firefox") as session: agent = client.Agent(session=session, reasoning_model='gemini/gemini-2.5-flash', max_steps=15) response = agent.run( task="Go to Hacker News (news.ycombinator.com) and extract the top 5 posts with their titles, URLs, points, authors, and comment counts.", response_format=TopPosts, ) print(response.answer)
Vaults are tools you can attach to your Agent instance to securely store and manage credentials. The agent automatically uses these credentials when needed.
from notte_sdk import NotteClient client = NotteClient() with client.Vault() as vault, client.Session(headless=False) as session: vault.add_credentials( url="https://x.com", username="your-email", password="your-password", ) agent = client.Agent(session=session, vault=vault, max_steps=10) response = agent.run( task="go to twitter; login and go to my messages", ) print(response.answer)
Personas are tools you can attach to your Agent instance to provide digital identities with unique email addresses, phone numbers, and automated 2FA handling.
from notte_sdk import NotteClient client = NotteClient() with client.Persona(create_phone_number=False) as persona: with client.Session(browser_type="firefox", headless=False) as session: agent = client.Agent(session=session, persona=persona, max_steps=15) response = agent.run( task="Open the Google form and RSVP yes with your name", url="https://forms.google.com/your-form-url", ) print(response.answer)
Stealth features include automatic CAPTCHA solving and proxy configuration to enhance automation reliability and anonymity.
from notte_sdk import NotteClient from notte_sdk.types import NotteProxy, ExternalProxy client = NotteClient() # Built-in proxies with CAPTCHA solving with client.Session( solve_captchas=True, proxies=True, # US-based proxy browser_type="firefox", headless=False ) as session: agent = client.Agent(session=session, max_steps=5) response = agent.run( task="Try to solve the CAPTCHA using internal tools", url="https://www.google.com/recaptcha/api2/demo" ) # Custom proxy configuration proxy_settings = ExternalProxy( server="http://your-proxy-server:port", username="your-username", password="your-password", ) with client.Session(proxies=[proxy_settings]) as session: agent = client.Agent(session=session, max_steps=5) response = agent.run(task="Navigate to a website")
File Storage allows you to upload files to a session and download files that agents retrieve during their work. Files are session-scoped and persist beyond the session lifecycle.
from notte_sdk import NotteClient client = NotteClient() storage = client.FileStorage() # Upload files before agent execution storage.upload("/path/to/document.pdf") # Create session with storage attached with client.Session(storage=storage) as session: agent = client.Agent(session=session, max_steps=5) response = agent.run( task="Upload the PDF document to the website and download the cat picture", url="https://example.com/upload" ) # Download files that the agent downloaded downloaded_files = storage.list(type="downloads") for file_name in downloaded_files: storage.download(file_name=file_name, local_dir="./results")
Cookies provide a flexible way to authenticate your sessions. While we recommend using the secure vault for credential management, cookies offer an alternative approach for certain use cases.
from notte_sdk import NotteClient import json client = NotteClient() # Upload cookies for authentication cookies = [ { "name": "sb-db-auth-token", "value": "base64-cookie-value", "domain": "github.com", "path": "/", "expires": 9778363203.913704, "httpOnly": False, "secure": False, "sameSite": "Lax" } ] with client.Session() as session: session.set_cookies(cookies=cookies) # or cookie_file="path/to/cookies.json" agent = client.Agent(session=session, max_steps=5) response = agent.run( task="go to nottelabs/notte get repo info", ) # Get cookies from the session cookies_resp = session.get_cookies() with open("cookies.json", "w") as f: json.dump(cookies_resp, f)
You can plug in any browser session provider you want and use our agent on top. Use external headless browser providers via CDP to benefit from Notte's agentic capabilities with any CDP-compatible browser.
from notte_sdk import NotteClient client = NotteClient() cdp_url = "wss://your-external-cdp-url" with client.Session(cdp_url=cdp_url) as session: agent = client.Agent(session=session) response = agent.run(task="extract pricing plans from https://www.notte.cc/")
Notte's close compatibility with Playwright allows you to mix web automation primitives with agents for specific parts that require reasoning and adaptability. This hybrid approach cuts LLM costs and is much faster by using scripting for deterministic parts and agents only when needed.
from notte_sdk import NotteClient client = NotteClient() with client.Session(headless=False, perception_type="fast") as session: # Script execution for deterministic navigation session.execute({"type": "goto", "url": "https://www.quince.com/women/organic-stretch-cotton-chino-short"}) session.observe() # Agent for reasoning-based selection agent = client.Agent(session=session) agent.run(task="just select the ivory color in size 6 option") # Script execution for deterministic actions session.execute({"type": "click", "selector": "internal:role=button[name=\"ADD TO CART\"i]"}) session.execute({"type": "click", "selector": "internal:role=button[name=\"CHECKOUT\"i]"})
Workflows are a powerful way to combine scripting and agents to reduce costs and improve reliability. However, deterministic parts of the workflow can still fail. To gracefully handle these failures with agents, you can use the AgentFallback
class:
import notte with notte.Session() as session: _ = session.execute({"type": "goto", "value": "https://shop.notte.cc/"}) _ = session.observe() with notte.AgentFallback(session, "Go to cart"): # Force execution failure -> trigger an agent fallback to gracefully fix the issue res = session.execute(type="click", id="INVALID_ACTION_ID")
For fast data extraction, we provide a dedicated scraping endpoint that automatically creates and manages sessions. You can pass custom instructions for structured outputs and enable stealth mode.
from notte_sdk import NotteClient from pydantic import BaseModel client = NotteClient() # Simple scraping response = client.scrape( url="https://notte.cc", scrape_links=True, only_main_content=True ) # Structured scraping with custom instructions class Article(BaseModel): title: str content: str date: str response = client.scrape( url="https://example.com/blog", response_format=Article, instructions="Extract only the title, date and content of the articles" )
Or directly with cURL
curl -X POST 'https://api.notte.cc/scrape' \ -H 'Authorization: Bearer <NOTTE-API-KEY>' \ -H 'Content-Type: application/json' \ -d '{ "url": "https://notte.cc", "only_main_content": false, }'
Search: We've built a cool demo of an LLM leveraging the scraping endpoint in an MCP server to make real-time search in an LLM chatbot - works like a charm! Available here: https://search.notte.cc/
This project is licensed under the Server Side Public License v1.
See the LICENSE file for details.
If you use notte in your research or project, please cite:
@software{notte2025, author = {Pinto, Andrea and Giordano, Lucas and {nottelabs-team}}, title = {Notte: Software suite for internet-native agentic systems}, url = {https://github.com/nottelabs/notte}, year = {2025}, publisher = {GitHub}, license = {SSPL-1.0} version = {1.4.4}, }
Copyright © 2025 Notte Labs, Inc.
33 contributors
giordano-lucas
@giordano-lucas
andreakiro
@andreakiro
leo-notte
@leo-notte
TonioSnowden
@TonioSnowden
shivankj11
@shivankj11
Gautam-Hegde
@Gautam-Hegde
omkar-notte
@omkar-notte
berkantay
@berkantay
NinoRisteski
@NinoRisteski
sam-notte
@sam-notte
coderabbitai[bot]
@coderabbitai