git clone https://github.com/borski/travel-hacking-toolkit
T=$(mktemp -d) && git clone --depth=1 https://github.com/borski/travel-hacking-toolkit "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/amex-travel" ~/.claude/skills/borski-travel-hacking-toolkit-amex-travel && rm -rf "$T"
skills/amex-travel/SKILL.mdAmex Travel Portal Search
Search the American Express travel portal for flights and hotels via Patchright. Returns cash prices, MR points pricing, International Airline Program (IAP) discounts, and Fine Hotels & Resorts / The Hotel Collection benefits.
Requires Patchright (undetected Playwright fork). Amex blocks standard Playwright and agent-browser.
Must run headed (headless=False). Amex detects headless browsers. On macOS, a Chrome window briefly appears. For background operation, use Docker.
Prerequisites
pip install patchright && patchright install chromium
Or use Docker (no local install needed):
docker pull ghcr.io/borski/amex-travel:latest # or build locally: docker build -t amex-travel skills/amex-travel/
When to Use
- Compare Amex portal MR pricing against cash and award prices
- Find IAP (International Airline Program) discounted fares on Platinum
- Find FHR and THC hotels with benefits ($100 credit, breakfast, upgrade)
- Compare portal redemption value against transfer-to-airline value
When NOT to Use
- Completing purchases. Find flights and hotels only. Do not book.
- Non-Platinum cards. IAP fares and FHR benefits require the Platinum Card.
Usage
Flight Search
# Local (opens a Chrome window briefly) python3 scripts/search_flights.py --origin SFO --dest CDG --depart 2026-08-11 # Round-trip business python3 scripts/search_flights.py --origin SFO --dest CDG --depart 2026-08-11 --return 2026-09-02 --cabin business # JSON output python3 scripts/search_flights.py --origin SFO --dest CDG --depart 2026-08-11 --json # Docker docker run --rm \ -v ~/.amex-travel-profiles:/profiles \ -e AMEX_USERNAME -e AMEX_PASSWORD \ amex-travel script /app/search_flights.py \ --origin SFO --dest CDG --depart 2026-08-11 --cabin business --json
Hotel Search
# Local python3 scripts/search_flights.py --hotel --dest "Oslo" --checkin 2026-08-13 --checkout 2026-08-15 # Docker docker run --rm \ -v ~/.amex-travel-profiles:/profiles \ -e AMEX_USERNAME -e AMEX_PASSWORD \ amex-travel script /app/search_flights.py \ --hotel --dest "Oslo" --checkin 2026-08-13 --checkout 2026-08-15 --json
Record Mode (API Discovery)
Capture network traffic during a manual search:
python3 scripts/search_flights.py --record
Offline Debug (Hotels)
Save and re-parse hotel results without re-running the browser:
# Save page HTML after hotel search python3 scripts/search_flights.py --hotel --dest "Paris" --checkin 2026-08-11 --checkout 2026-08-15 --save-html /tmp/amex-hotels.json # Re-parse locally (instant, no browser) python3 scripts/search_flights.py --parse-html /tmp/amex-hotels.json
2FA Flow
Amex uses email OTP for 2FA. After first login with "Add This Device", subsequent runs skip 2FA from the same profile.
How it works: When 2FA is triggered, the script prints
2FA_CODE_NEEDED to stdout and 2FA REQUIRED to stderr, then polls for the code. It will wait up to 2 minutes.
For agents: When you see
2FA_CODE_NEEDED in the script output, ask the user for the verification code Amex just emailed them. Once they provide it, write it to the code file:
echo "123456" > /tmp/amex-2fa-code.txt
The script picks up the file automatically and continues login.
Command hook (optional, for full automation): Set
AMEX_2FA_COMMAND to a command that blocks until it has the code, then prints it to stdout. The script runs this instead of polling the file.
After first login with "Add This Device", 2FA is skipped on repeat runs from the same profile.
How It Works
Flight Search Architecture
- Auth: Cookie injection from saved profile. Falls back to fresh login with email 2FA.
- Form filling: DOM-based search form automation (airport autocomplete, calendar picker, cabin selector)
- Login gate: After form submission, Amex redirects through a login interstitial. Script handles re-authentication automatically.
- Data extraction: Flight results are server-side rendered into
(a 627KB Redux store). Script parses this JSON blob for all flight data.window.appData - IAP detection: Platinum Card holders see
(IAP) fare types alongsidePEP
(public) fares. IAP fares are typically 10-15% cheaper for front-of-cabin international flights.PUB
Hotel Search Architecture
- Form filling: Same DOM-based approach as flights
- Login gate: Handled automatically
- Data extraction: Hotels render as a Next.js app with NO
. Script parses the DOM usingwindow.appData
elements.data-testid="hotel-offer-card" - FHR/THC detection: Identified via
text ("Fine Hotels and Resorts" or "The Hotel Collection")data-testid="offer-banner" - Benefits extraction: FHR/THC cards show benefits (breakfast, credit, upgrade) as
elementsdata-testid="offer-amenities-item"
Data Structure
Flight results (from
window.appData.flightSearch.itineraries[]):
withpricing_information[]
=fare_type
(IAP) orPEP
(public)PUB
(cash),total_price.cents
(MR points = 1 cent per point)total_price_in_points
with carrier, times, duration, cabin, equipment, amenitiessegment.legs[]
,segment.seats_left
,is_refundablecancellation_policy
Hotel results (from DOM parsing):
- Hotel name, stars, city, distance
- TripAdvisor rating and review count
- Per-night price and total price
- MR points cost
- FHR/THC membership with specific benefits
- Standard amenities (wifi, breakfast, parking)
International Airline Program (IAP)
Platinum Card benefit. Lower fares on premium cabin seats for international flights on select airlines. Shows as a separate
PEP fare type alongside PUB (public fare).
- Typically 10-15% savings on business/first class
- Not available on all routes or airlines
- Only visible when logged in with a Platinum Card
Output Format
Always use markdown tables.
Flights
| # | Airline | Route | Stops | Duration | Cash | IAP Cash | Points | Seats |
|---|---|---|---|---|---|---|---|---|
| 1 | Turkish | SFO-IST-CDG | 1 | 20h 10m | $5,044 | $4,381 | 438,113 | 3 |
Hotels
| # | Hotel | Program | Stars | Per Night | Total | Points | Benefits |
|---|---|---|---|---|---|---|---|
| 1 | Hotel Continental | FHR | 5 | $471 | $942 | 94,200 | Breakfast, $100 credit, upgrade, 4pm checkout |
After Tables
- Flag IAP savings (show % discount)
- Note FHR/THC benefits and how they offset the rate
- Calculate effective CPP for MR redemptions (1 point = 1 cent at Amex portal)
- Compare against transfer-to-airline value
- Mention the $600/yr Platinum hotel credit ($300 per half-year, shared between FHR and THC)
Cabin Codes
| CLI Value | Amex Code | Description |
|---|---|---|
| | Standard economy |
| | Premium economy |
| | Business class |
| | First class |
Environment Variables
| Variable | Required | Description |
|---|---|---|
| Yes | Amex online account username |
| Yes | Amex online account password |
| No | Browser profile directory (default: ) |
| No | Command that blocks until email code is ready, prints to stdout |
Troubleshooting
- Login gate after search: Normal. Amex always redirects through a login interstitial after form submission. The script handles this automatically.
- No appData found (flights): The page may not have fully loaded. Script waits for the Redux store to populate. Check if login succeeded.
- Empty hotel results: Hotels use DOM parsing, not appData. If the DOM structure changed, the
selectors may need updating.data-testid - Calendar picker fails: Amex uses
for calendar days (notdiv[role="button"]
). The script uses class patterns<button>
to find the right month container.automation-date-picker-month-{year}-{month} - 2FA code rejected: Amex codes expire quickly. Make sure the code is fresh (not an old one from a previous login).
Limitations
- Headed mode required. Amex detects headless. Docker+xvfb is the workaround.
- ~45 seconds per search. Login + form fill + login gate + results load.
- Hotel results via DOM only. No API interception available for hotels (Next.js app with empty
). Parser depends on__NEXT_DATA__
attributes.data-testid - Device trust helps. After "Add This Device" on first login, 2FA is skipped for that profile. Keep profiles persistent via Docker volume mounts.