Google Hotels MCP server and Python library โ€” reverse-engineered batchexecute client with FastMCP tools

๐Ÿจ stays โ€” Google Hotels MCP Server + Python Library

CIPyPIPythonLicense: MIT

A single Python package that gives you Google Hotels three ways: a CLI, anMCP server for Claude / Codex / ChatGPT, and an importable library.All three talk directly to Google's internal batchexecute RPC โ€” no HTMLscraping, no headless browser, no unofficial proxies.

๐Ÿš€ Why stays?

  • Fast โ€” direct RPC calls, not page rendering
  • Zero scraping โ€” no HTML parsing, no Playwright/Puppeteer at runtime
  • Reliable โ€” Chrome TLS impersonation via curl_cffi, 10 rps rate-limit bucket, tenacity retries
  • MCP-native โ€” three tools, two prompts, one resource; stdio and streamable HTTP
  • One install, three surfaces โ€” pipx install stays gets you the CLI, the MCP server, and the library

Quick start

Don't want to touch the terminal? If you already use Claude Code orCodex, paste this into the chat and your assistant will handleeverything โ€” uv, the stays install, and the MCP client registration:

Install the stays MCP server on this Mac and register it with your CLI. Use uv (install it from https://astral.sh/uv/install.sh if it's not already there) โ€” don't use Homebrew. Once stays is installed, run stays setup <your-host> (use claude if you're Claude Code, codex if you're Codex) and tell me to restart this session.

Then restart your CLI and ask for a hotel search.

Prefer the terminal?

# Install
pipx install stays

# Register the MCP server with whichever client(s) you use
stays setup claude        # Claude Code CLI and/or Claude Desktop
stays setup codex         # OpenAI Codex CLI
stays setup chatgpt       # Instructions for remote HTTPS + Developer Mode

# Or skip MCP and use the CLI directly
stays "tokyo hotels" --check-in 2026-07-22 --check-out 2026-07-26

Restart your MCP client, then try:

"Find me a 4-star hotel in Tokyo for July 22โ€“26 under $120 a night."

"Compare rooms, rates, and cancellation for the top 5 hotels near Big Ben."

"Show me pet-friendly refundable stays in Paris for next weekend."

Prefer a different install path? See Install below.

Pick your path

You areโ€ฆ Start here
A Claude / Codex / ChatGPT user who wants your assistant to search hotels MCP Clients โ†’ MCP Tools
Running hotel searches from the terminal CLI Usage
Building a Python app on top of Google Hotels Python API
An AI coding agent installing stays for a user For AI Agents
Deploying stays as an HTTP MCP server Running the server directly โ†’ Docker

Features

  • ๐Ÿ” List-view search โ€” 16 filter slots: city / brand / stars / price range / amenities / dates / guests / cancellation / eco / special offers / sort.
  • ๐Ÿจ Deep hotel detail โ€” rooms, per-OTA rate plans (Booking, Expedia, Hotels.com, Trip.com, direct), cancellation policies, deep-link URLs.
  • โšก Parallel enrichment โ€” search + fan-out detail fetch for the top N hotels in a single call, with per-hotel partial failure.
  • ๐Ÿค– MCP server โ€” FastMCP over stdio (what Claude/Codex spawn) or streamable HTTP (dev / Docker).
  • ๐Ÿงฐ Three-format CLI โ€” text (rich tables), json (single envelope), jsonl (stream-friendly).
  • ๐Ÿ›ก๏ธ Production hygiene โ€” rate-limited curl_cffi session with Chrome TLS impersonation, tenacity exponential backoff, typed pydantic v2 models, 330 offline tests.
  • ๐Ÿณ Ready for containers โ€” published multi-arch image at ghcr.io/him229/stays:latest, plus docker-compose profiles.

Install

# Recommended โ€” isolated venv, `stays` on your PATH
pipx install stays

# Inside an existing environment
pip install stays

# From source (latest main)
pip install 'git+https://github.com/him229/stays.git'

# Local dev checkout
git clone https://github.com/him229/stays.git
cd stays
uv sync --extra dev
uv run stays --help

Requires Python 3.10+. There are no optional extras โ€” the CLI, the MCPstdio/HTTP server, and the Python library are all included in the single coreinstall.

CLI Usage

The stays console script is the only entry point you need. Subcommands:

Command Purpose
stays search <query> Fast list-view search (one RPC)
stays details <entity_key> Rooms / rates / cancellation for ONE hotel
stays enrich <query> Search + parallel detail fetch for the top N hotels
stays mcp Stdio MCP server (what Claude / Codex spawn)
stays mcp-http Streamable-HTTP MCP server (dev / Docker)
stays setup {claude|codex|chatgpt} Register the MCP server with a client

Smart default: if the first positional arg doesn't match a knownsubcommand, stays routes to search. stays "paris hotels" ... isequivalent to stays search "paris hotels" ....

Examples

# Rich list-view with filters
stays search "tokyo hotels" \
    --check-in 2026-07-22 --check-out 2026-07-26 \
    --stars 4 --stars 5 \
    --amenity POOL --brand HILTON \
    --price-max 300 --sort-by LOWEST_PRICE

# Smart-default form (no `search` subcommand)
stays "paris hotels" --check-in 2026-09-01 --check-out 2026-09-04

# Rooms / rates / cancellation for ONE hotel
stays details "ChkI_ENTITY_KEY_FROM_SEARCH" \
    --check-in 2026-07-22 --check-out 2026-07-26

# Search + top-5 deep detail in parallel
stays enrich "new york hotels" --max-hotels 5 \
    --check-in 2026-09-01 --check-out 2026-09-04

# Machine-readable output
stays search "tokyo" --format json    # single pretty-printed envelope
stays search "tokyo" --format jsonl   # one record per line, stream-friendly

CLI options (search / enrich)

Flag Type Purpose
--check-in / --check-out YYYY-MM-DD Stay window (required for rate plans)
--adults / --children int Party composition (1โ€“12 / 0โ€“8)
--child-age int (repeat) One --child-age per child
--currency ISO 4217 Output currency (default USD)
--property-type enum HOTELS (default) or VACATION_RENTALS
--sort-by enum RELEVANCE, LOWEST_PRICE, HIGHEST_RATING, MOST_REVIEWED
--stars 1โ€“5 (repeat) Hotel-class filter (--stars 4 --stars 5)
--min-rating enum THREE_FIVE_PLUS, FOUR_ZERO_PLUS, FOUR_FIVE_PLUS
--amenity enum (repeat) POOL, WIFI, SPA, PET_FRIENDLY, โ€ฆ
--brand enum (repeat) HILTON, MARRIOTT, HYATT, โ€ฆ
--price-min / --price-max int Price band (selected currency)
--free-cancellation flag Refundable-only
--eco-certified flag Eco-certified only
--special-offers flag Deals only
--max-results int search only โ€” cap (1โ€“25)
--max-hotels int enrich only โ€” cap (1โ€“15, default 5)
--format enum text (rich tables, default), json, jsonl

--format json / --format jsonl envelope shapes are stable for v0.1.xbut may evolve in minor releases.

MCP Clients

One command per client. If auto-registration isn't possible, each backendprints the equivalent JSON/TOML you can paste yourself.

Claude Code / Desktop

stays setup claude

Auto-detects both the claude CLI (Claude Code) andclaude_desktop_config.json (Claude Desktop) and registers with whichever itfinds. Falls through to printing the canonical JSON when neither is present.

  • --print-json โ€” always print, never register.
  • --desktop-only โ€” skip the claude CLI probe and force Desktop mode.
  • --replace โ€” overwrite any prior stays entry.

Codex CLI

stays setup codex

Shells to codex mcp add stays -- <abs-path>/stays mcp when the codexbinary is on $PATH; otherwise prints the equivalent TOML block for~/.codex/config.toml.

  • --print-toml โ€” always print, never shell out.
  • --replace โ€” overwrite any prior stays entry.

ChatGPT

stays setup chatgpt

Prints setup instructions. ChatGPT requires a public HTTPS endpointimplementing OAuth 2.1 + Dynamic Client Registration, registered viaDeveloper Mode in the ChatGPT app โ€” no local auto-registration is possible.

  • --open โ€” jump to the ChatGPT Connectors settings page in your browser.

Canonical MCP client config

If the stays setup โ€ฆ installer cannot detect your client, emit the snippetyourself:

stays setup claude --print-json

A minimal version that works when the stays binary is on the client's$PATH:

{
  "mcpServers": {
    "stays": {
      "command": "/abs/path/to/stays",
      "args": ["mcp"]
    }
  }
}

Claude Desktop config path:

OS Path
macOS ~/Library/Application Support/Claude/claude_desktop_config.json
Linux ~/.config/Claude/claude_desktop_config.json
Windows %APPDATA%\Claude\claude_desktop_config.json

MCP Tools

The server exposes three tools. All of them return JSON-safe dicts.

Tool When to use RPC cost
search_hotels List-view discovery: browse / filter by city, stars, amenities, price, brand. Start here. 1
get_hotel_details One hotel: rooms, per-OTA rates, cancellation. Needs an entity_key from search_hotels. 1
search_hotels_with_details Compare 3โ€“15 hotels' rooms/rates/cancellation in a single call. 1 + N

search_hotels parameters

Parameter Type Description
query required string "tokyo hotels", "Hilton Paris", etc.
check_in / check_out string YYYY-MM-DD. Omit both for flexible dates.
adults / children / child_ages int / int / list[int] Party composition
currency string ISO 4217 (default from STAYS_MCP_DEFAULT_CURRENCY)
property_type enum HOTELS (default) or VACATION_RENTALS
sort_by enum RELEVANCE, LOWEST_PRICE, HIGHEST_RATING, MOST_REVIEWED
hotel_class list[int] Star classes to include, e.g. [4, 5]
min_guest_rating enum THREE_FIVE_PLUS, FOUR_ZERO_PLUS, FOUR_FIVE_PLUS
amenities list[string] POOL, WIFI, SPA, PET_FRIENDLY, โ€ฆ
brands list[string] HILTON, MARRIOTT, HYATT, IHG, ACCOR, โ€ฆ
free_cancellation bool Refundable-only
eco_certified bool Eco-certified only
special_offers bool Deals only
price_min / price_max int Price band (selected currency)
max_results int Cap (1โ€“25); overrides STAYS_MCP_MAX_RESULTS

get_hotel_details parameters

Parameter Type Description
entity_key required string From a prior search_hotels result
check_in required string YYYY-MM-DD (rate plans are date-keyed)
check_out required string YYYY-MM-DD after check_in
currency string ISO 4217 (default USD)

search_hotels_with_details parameters

Same filter set as search_hotels, plus:

Parameter Type Description
max_hotels int Top-N hotels to enrich (1โ€“15, default 5)

Prompts & resources

The server also exposes two prompts โ€” when-to-deep-search andcompare-hotels-in-city โ€” that help an LLM pick the right tool, plus oneresource resource://stays-mcp/configuration describing the live env-varconfig.

Python API

Everything public is re-exported from the top-level stays package.

from datetime import date
from stays import (
    SearchHotels, HotelSearchFilters, Location, DateRange, GuestInfo,
    Amenity, Brand, Currency, SortBy, MinGuestRating,
)

s = SearchHotels()

# 1. Fast list-view search โ€” one RPC
results = s.search(HotelSearchFilters(
    location=Location(query="tokyo hotels"),
    dates=DateRange(check_in=date(2026, 7, 22), check_out=date(2026, 7, 26)),
    guests=GuestInfo(adults=2),
    hotel_class=[4, 5],
    amenities=[Amenity.POOL, Amenity.WIFI],
    brands=[Brand.HILTON],
    sort_by=SortBy.LOWEST_PRICE,
    currency=Currency.USD,
))
for hotel in results[:3]:
    print(hotel.name, hotel.display_price, hotel.overall_rating)

Deep detail for one hotel

first = results[0]
if first.entity_key:
    detail = s.get_details(
        entity_key=first.entity_key,
        dates=DateRange(check_in=date(2026, 7, 22), check_out=date(2026, 7, 26)),
    )
    print(detail.address, detail.phone)
    for room in detail.rooms:
        for rp in room.rates:
            print(rp.provider, rp.price, rp.cancellation.kind.value)

Parallel enrichment with partial-failure handling

filters = HotelSearchFilters(
    location=Location(query="new york hotels"),
    dates=DateRange(check_in=date(2026, 9, 1), check_out=date(2026, 9, 4)),
)
for item in s.search_with_details(filters, max_hotels=5):
    if item.ok:
        print(item.detail.name, len(item.detail.rooms), "rooms")
    else:
        # error_kind is "transient" or "fatal"; is_retryable is True only
        # for transient failures. Unknown exceptions (parser bugs, etc.)
        # propagate โ€” only typed BatchExecuteError / TransientBatchExecuteError
        # / MissingHotelIdError become per-item errors.
        retry_hint = " (retryable)" if item.is_retryable else ""
        print("skipped:", item.result.name, "โ€”", item.error_kind, item.error, retry_hint)

stays enrich --format json and the MCP search_hotels_with_details toolmirror this shape: each per-hotel record includes ok, result, detail,error, error_kind ("transient" | "fatal" | null), and is_retryable.

Serializer-only (no HTTP)

Useful for debugging the wire shape or building your own client on top:

filters = HotelSearchFilters(
    location=Location(query="new york hotels"),
    dates=DateRange(check_in=date(2026, 9, 1), check_out=date(2026, 9, 4)),
    guests=GuestInfo(adults=2, children=1, child_ages=[7]),
    price_range=(100, 300),
)
filters.format()           # Python list โ€” inner JSON shape
filters.encode()           # URL-encoded outer envelope
filters.to_request_body()  # "f.req=..." โ€” ready to POST

Public exports

  • Models: Amenity, Brand, Currency, DateRange, GuestInfo,HotelSearchFilters, Location, MinGuestRating, PropertyType, SortBy
  • Results: HotelResult, HotelDetail, RoomType, RatePlan,CancellationPolicy, CancellationPolicyKind, Review,RatingHistogram, CategoryRating, NearbyPlace, EnrichedResult(now carries error_kind: Literal["transient","fatal"] | None anda .is_retryable property)
  • Search API: SearchHotels, Client, BatchExecuteError,TransientBatchExecuteError, MissingHotelIdError
  • Serializers: stays.serialize โ€” canonical serialize_hotel_result,serialize_hotel_detail, plus build_success / build_error envelopehelpers (shared by CLI + MCP; dict shapes guarded by golden-fixture tests)
  • MCP (core install only): mcp, search_hotels, get_hotel_details,search_hotels_with_details, run_mcp, run_mcp_http

Running the server directly

# Stdio โ€” what Claude Code / Desktop / Codex invoke on your behalf
stays mcp

# Streamable HTTP โ€” dev or Docker runtime
stays mcp-http   # serves http://127.0.0.1:8000/mcp/

The streamable-HTTP endpoint requires the MCP-spec headerAccept: application/json, text/event-stream. A bare GET /mcp/ returns405/406 by design โ€” this is not a bug.

Docker

A published image is available from GitHub Container Registry:

# Pull the latest release image
docker run --rm -p 8000:8000 ghcr.io/him229/stays:latest

# Or with compose (prod profile, healthcheck included)
docker compose --profile prod up

# Or build + run locally (dev profile)
docker compose --profile dev up --build

Environment variables (see Configuration) are passed throughnormally, e.g. -e STAYS_RPS=5 -e STAYS_MCP_DEFAULT_CURRENCY=EUR.

Configuration

All configuration is via environment variables. STAYS_RPS tunes the sharedrate-limit bucket used by the library, CLI, and MCP server alike; everythingelse is prefixed STAYS_MCP_ and only affects MCP tool defaults.

Env var Default Purpose
STAYS_RPS 10 Rate-limiter throttle (requests per second)
STAYS_MCP_DEFAULT_ADULTS 2 Default adults per search
STAYS_MCP_DEFAULT_CURRENCY USD Fallback currency
STAYS_MCP_DEFAULT_SORT_BY RELEVANCE Default sort
STAYS_MCP_MAX_RESULTS unset Cap on returned list-view results (uncapped when unset)
STAYS_MCP_DEFAULT_MAX_HOTELS_WITH_DETAILS 5 Default N for search_hotels_with_details (hard cap 15)

The live config resource is also readable atresource://stays-mcp/configuration from the running MCP server.

For AI Agents

If you are an AI agent installing this on behalf of a human user, usethe commands in this section verbatim. They are the canonical install path.pipx install stays is always sufficient โ€” there are no optional[mcp] / [cli] extras to remember.

# Option A (recommended) โ€” pipx: isolated venv + `stays` on PATH
pipx install stays
stays setup claude        # registers with any Claude client detected

# Option B โ€” inside an existing Python environment
pip install stays

# Option C โ€” local dev checkout
git clone https://github.com/him229/stays.git
cd stays
uv sync --extra dev
uv run stays setup claude

Full agent-facing docs โ€” verification steps, troubleshooting table, scriptedinstall โ€” live in docs/AI_AGENTS.md.

Development

git clone https://github.com/him229/stays.git
cd stays
make install-dev          # uv sync --extra dev
make test                 # offline suite
make test-live            # live Google-API tests (network + rate-limit)
make lint                 # ruff check
make format               # ruff format
make mcp                  # run stdio MCP server locally
make mcp-http             # run streamable-HTTP MCP server on :8000
make coverage             # pytest + branch coverage โ†’ htmlcov/
make build                # sdist + wheel

Full command list: make help.

Testing

  • Offline suite (make test) โ€” 330 tests including golden-fixtureregression guards for the parser, the canonical serializers, and the CLIJSON envelope shapes (tests/test_parse_golden.py,tests/test_serialize_golden.py, tests/test_cli_envelope_golden.py).
  • Live CLI E2E (tests/test_cli_live.py, marker-gated: pytest -m live)โ€” 9 subprocess-driven scenarios that exercise the real stays binaryagainst live Google (Tokyo dates, Hilton brand family, 4/5-star Paris,London amenity + price band, free-cancellation differential andrefundability, searchโ†’details roundtrip, enrich parallel per-itemcontract, JPY sort).
  • Browser-verify matrix (pytest --browser-verify) โ€” theMCP-vs-browser and CLI-vs-browser oracle suites undertests/browser_verification/ (includingtests/browser_verification/test_cli_vs_browser.py) diff our resultsagainst an authoritative browser oracle. The driver is pluggable:agent-browser is the default; set STAYS_BROWSER_DRIVER=playwrightto force the Playwright fallback.

Project layout

stays/
โ”œโ”€โ”€ cli/           # typer app + subcommand bodies
โ”œโ”€โ”€ mcp/           # FastMCP server, per-client setup backends
โ”œโ”€โ”€ search/        # batchexecute HTTP client + search / detail / enrich
โ””โ”€โ”€ models/        # pydantic v2 filter + result + detail + policy models

tests/             # 330 offline + live (-m live) + browser-verify (--browser-verify)
docs/              # reverse-engineering notes, superpowers artifacts, AI_AGENTS.md
captures/          # Playwright capture oracles (gitignored where large)

Contributing

Contributions welcome โ€” see CONTRIBUTING.md for the devloop, PR checklist, and issue template.

Acknowledgements

Inspired by punitarani/fli, whichdemonstrated the same direct batchexecute approach for Google Flightsโ€” a cleaner path than HTML scraping or headless-browser automation.stays applies that pattern to Google Hotels.

License

MIT โ€” see LICENSE.

MCP Server ยท Populars

MCP Server ยท New

    punkpeye

    FastMCP

    A TypeScript framework for building MCP servers.

    Community punkpeye
    can4hou6joeng4

    boss-agent-cli

    AI-agent-first CLI for BOSS ็›ด่˜ โ€” ่Œไฝๆœ็ดขใ€็ฆๅˆฉ็ญ›้€‰ใ€ๆ‹›่˜่€…ๅทฅไฝœๆตใ€MCP ๅทฅๅ…ทไธŽ AI ็ฎ€ๅކไผ˜ๅŒ–

    Community can4hou6joeng4
    clidey

    WhoDB

    A lightweight next-gen data explorer - Postgres, MySQL, SQLite, MongoDB, Redis, MariaDB, Elastic Search, and Clickhouse with Chat interface

    Community clidey
    Battam1111

    Myco

    Self-evolving cognitive organism for AI agents โ€” eternal devouring, eternal evolution.

    Community Battam1111
    MLS-Tech-Inc

    Shortlist MCP Server

    MCP server for Shortlist โ€” search, queue, and auto-apply to jobs from Claude Code

    Community MLS-Tech-Inc