Travel-hacking-toolkit seats-aero
Search award flight availability across mileage programs via Seats.aero Partner API. Use when planning trips on points, searching for award flights, or comparing mileage costs across programs. Triggers on "award flight", "miles cost", "points for flights", "cheapest award", "seats.aero", "mileage program availability", "United miles", "Flying Blue", "Aeroplan", or any question about how many miles/points a specific flight costs.
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/seats-aero" ~/.claude/skills/borski-travel-hacking-toolkit-seats-aero && rm -rf "$T"
skills/seats-aero/SKILL.mdSeats.aero Skill
Search cached and live award flight availability across 27 mileage programs (including Spirit Free Spirit and Frontier Miles, added April 2026). Find the cheapest award flights, compare programs, and get booking links.
Source: seats.aero | API Docs | Knowledge Base
Authentication
Set
SEATS_AERO_API_KEY in your .env file. Get from Settings > API tab (Pro account required).
All requests use the
Partner-Authorization header. Pro users get 1,000 API calls per day for non-commercial use.
API Base
https://seats.aero/partnerapi
Key Concepts
Saver vs Dynamic Pricing
Award seats fall into two categories:
- Saver (fixed price): Set, predictable miles cost. Often the ONLY type bookable through partner programs. Best value. Always shown on Seats.aero.
- Dynamic: Price changes based on demand (like cash fares). Can be reasonable or extremely expensive. Seats.aero filters out overpriced dynamic awards by default.
Dynamic Price Filter Thresholds
Awards exceeding these per-hour thresholds are hidden automatically:
| Program | Economy | Prem Econ | Business | First |
|---|---|---|---|---|
| American | <100K total | <100K total | <150K total | <200K total |
| United | 7,500/hr | 10,000/hr | 14,000/hr | 20,000/hr (max 10h) |
| Delta | 10,000/hr | 15,000/hr | 20,000/hr | N/A |
| Aeroplan | 7,500/hr | 10,000/hr | 12,500/hr | 15,000/hr |
| Flying Blue | 5,000/hr | 10,000/hr | 15,000/hr | 15,000/hr |
| Alaska | <150K total | <150K total | <150K total | <150K total |
| Aeromexico | 10,000/hr (max 10h) | 15,000/hr | 20,000/hr | N/A |
| Etihad | 10,000/hr (max 10h) | 15,000/hr | 20,000/hr | N/A |
| JetBlue | 10,000/hr | 15,000/hr | 20,000/hr | N/A |
| Qantas | 7,500/hr | 10,000/hr | 20,000/hr | 25,000/hr |
To see ALL dynamic pricing (including expensive): pass
include_filtered=true on cached search/bulk, or disable_filters=true / show_dynamic_pricing=true on live search.
Phantom Availability
Sometimes a flight shows bookable but no seat actually exists. Causes:
- Cached/stale results: Check
for freshness.ComputedLastSeen - Syncing delays: Seats taken but partner systems haven't updated.
- Partner feed glitches: Programs incorrectly show space on partners.
Always verify on the airline's own site before transferring points. Compare multiple programs in the same alliance. Call to confirm and hold if possible.
Foreign Program Caveats
Some programs (Smiles, TudoAzul, etc.) may require local tax IDs, local payment methods, or are only available in the local language. They still appear in results because they can serve as signals: if Azul TudoAzul shows United saver space, that same space is likely bookable through Aeroplan or other Star Alliance partners you CAN access.
Award Release Dates
Airlines release award seats at different advance windows (e.g., Aeroplan releases up to 358 days out). Check seats.aero/tools/releases to see the maximum days out for each airline by program. Useful for planning when to start searching.
Availability Objects
Each result from Cached Search / Bulk Availability is a summary that groups all flights for one route/date/program into a single object. For example, if Alaska shows 3 different SFO>LAX flights on Mar 16, they all appear in one availability object. The
MileageCost fields show the cheapest option. The RemainingSeats fields show the maximum across all flights. Use Get Trips to see individual flight details.
Mileage Program Sources
| Source | Program | Cabins | Seat Count | Trip Data | Taxes |
|---|---|---|---|---|---|
| SAS EuroBonus | Y/J | Yes | Yes | Yes |
| Virgin Atlantic Flying Club | Y/W/J | Yes | Yes | Yes |
| Aeromexico Club Premier | Y/W/J | Yes | Yes | Yes |
| American Airlines AAdvantage | Y/W/J/F | Low seats only* | Yes | Yes |
| Delta SkyMiles | Y/W/J | Yes | Yes | Yes |
| Etihad Guest | Y/J/F | Yes | Yes | Yes |
| United MileagePlus | Y/W/J/F | Yes | Yes | Yes |
| Emirates Skywards | Y/W/J/F | No | Partial** | Yes |
| Air Canada Aeroplan | Y/W/J/F | Usually*** | Yes | Yes |
| Alaska Mileage Plan | Y/W/J/F | Yes | Yes | Yes |
| Virgin Australia Velocity | Y/W/J/F | Yes | Yes | Yes |
| Qantas Frequent Flyer | Y/W/J/F | No | Yes | Yes |
| Copa ConnectMiles | Y/J/F | No | Yes | Yes |
| Azul TudoAzul | Y/J | No | Yes | Yes |
| GOL Smiles | Y/W/J/F | Yes | Yes | Yes |
| Air France/KLM Flying Blue | Y/W/J/F | Yes | Yes | Yes |
| JetBlue TrueBlue | Y/W/J/F | Yes | Yes | Yes |
| Qatar Privilege Club | Y/J/F | No | Yes | No |
| Turkish Miles & Smiles | Y/J | No | Yes | No |
| Singapore KrisFlyer | Y/W/J/F | No | Yes | No |
| Ethiopian ShebaMiles | Y/J | Yes | Yes | Yes |
| Saudi AlFursan | Y/J/F | Yes | Yes | Yes |
| Finnair Plus | Y/W/J/F | Yes | Yes | Yes |
| Lufthansa Miles&More | Y/J/F | Yes | Yes | Yes |
| Frontier Airlines | Y | No | Yes | Yes |
| Spirit Airlines | Y | Yes | Yes | Yes |
* AA provides seat counts only when remaining seats are low. Otherwise shows 0. ** Emirates connection flights may have missing fields. *** Aeroplan seat count is typically available but may be 0 in rare cases.
Cabin codes: Y = economy, W = premium economy, J = business, F = first
Cached Search (Primary Endpoint)
Search for award availability between specific airports and date ranges across all programs. Use this for targeted route/date searches.
curl -s -H "Partner-Authorization: $SEATS_AERO_API_KEY" \ "https://seats.aero/partnerapi/search?origin_airport=SFO&destination_airport=NRT&start_date=2026-08-01&end_date=2026-08-31" | jq '.'
Parameters
| Param | Required | Description |
|---|---|---|
| Yes | Comma-delimited airport codes: |
| Yes | Comma-delimited airport codes: |
| No | YYYY-MM-DD format |
| No | YYYY-MM-DD format |
| No | Comma-delimited: |
| No | Comma-delimited program filter: |
| No | Boolean. Only direct flights. |
| No | Comma-delimited airline codes: |
| No | Default: date+cabin priority. for cheapest first. |
| No | Boolean. Include flight details in response (slower, larger). |
| No | Boolean. When combined with , returns fewer fields per trip for better performance. |
| No | Boolean. Show expensive dynamic awards that were filtered out. |
| No | Results per page. 10-1000, default 500. |
| No | Pagination offset. |
| No | Cursor from previous response for consistent pagination. |
Response Fields (Availability Object)
Each result summarizes all flights for one route/date/program:
| Field | Description |
|---|---|
| Availability ID (use with for flight details) |
| Origin airport code |
| Destination airport code |
| Departure date |
| Mileage program |
, , , | Cabin availability booleans |
, , , | Cheapest points cost across all flights (string) |
, , , | Max seats remaining across all flights |
, , , | Operating carriers |
, , , | Direct flight available |
| When availability was last verified |
Get Trip Details
Get flight-level information from an availability object. Returns individual itineraries with segments, booking links, and coordinates.
curl -s -H "Partner-Authorization: $SEATS_AERO_API_KEY" \ "https://seats.aero/partnerapi/trips/{availability_id}" | jq '.'
Parameters
| Param | Required | Description |
|---|---|---|
| Yes (path) | The availability object ID |
| No (query) | Include expensive dynamic results filtered out |
Response Fields (Trip Object)
| Field | Description |
|---|---|
| economy, premium, business, first |
| Points cost (integer) |
| Cost through alliance partner programs (integer) |
| Taxes in cents (divide by 100 for dollars) |
| Currency code (empty = USD) |
| Seats left |
| Number of stops |
| Minutes |
| Operating airlines |
| Flight number string |
| Departure (local airport time) |
| Arrival (local airport time) |
| Array of individual flight legs |
| Mileage program |
The response also includes:
withbooking_links[]
,label
, andlink
boolean for direct booking URLsprimary
andorigin_coordinates
(Lat/Lon)destination_coordinates
Trip Segment Fields
Each segment in
AvailabilitySegments:
| Field | Description |
|---|---|
| e.g., "TK800" |
| e.g., "77W" |
| e.g., "77W" |
| Segment origin |
| Segment destination |
| Departure time (local airport) |
| Arrival time (local airport) |
| Booking class letter (e.g., "I", "X") |
| Segment distance in miles |
| Segment order (0-indexed) |
Bulk Availability
Retrieve large result sets from one mileage program. Use for broad regional exploration (e.g., "all United business class from North America to Europe").
curl -s -H "Partner-Authorization: $SEATS_AERO_API_KEY" \ "https://seats.aero/partnerapi/availability?source=united&origin_region=North%20America&destination_region=Europe&cabin=business&start_date=2026-08-01&end_date=2026-09-30" | jq '.'
Parameters
| Param | Required | Description |
|---|---|---|
| Yes | Single mileage program |
| No | economy, premium, business, first |
| No | YYYY-MM-DD |
| No | YYYY-MM-DD |
| No | North America, South America, Africa, Asia, Europe, Oceania |
| No | Same as above |
| No | Boolean. Show filtered-out dynamic pricing. |
| No | 10-1000, default 500 |
| No | Pagination offset |
| No | From previous response |
Get Routes
List all monitored routes for a mileage program. Useful for understanding what city pairs have cached data.
curl -s -H "Partner-Authorization: $SEATS_AERO_API_KEY" \ "https://seats.aero/partnerapi/routes?source=united" | jq '.'
Returns an array of route objects:
| Field | Description |
|---|---|
| Origin code |
| Destination code |
| Region name |
| Region name |
| How far ahead this route is scanned |
| Route distance |
| Mileage program |
Live Search
Real-time search for ANY city pair (not limited to monitored routes). Commercial agreement required. Not available to Pro users via API.
5-15 second response times. Build proper error handling with exponential backoff.
curl -s -X POST -H "Partner-Authorization: $SEATS_AERO_API_KEY" \ -H "Content-Type: application/json" \ -d '{"origin_airport":"SFO","destination_airport":"NRT","departure_date":"2026-08-15","source":"united","seat_count":2}' \ "https://seats.aero/partnerapi/live" | jq '.'
Parameters (JSON body)
| Param | Required | Description |
|---|---|---|
| Yes | Single airport code |
| Yes | Single airport code |
| Yes | YYYY-MM-DD |
| Yes | Single mileage program |
| No | 1-9, default 1 |
| No | Disable ALL filters (dynamic pricing + mismatched airports) |
| No | Disable only dynamic pricing filter, keep airport filtering |
Important: IDs returned from live search are NOT real and cannot be used with other Seats.aero APIs (Get Trips, etc.). Live results are ephemeral.
Pagination
All list endpoints use
skip + cursor. On first call, omit both. From the response, save the cursor value. On subsequent calls, pass cursor and increment skip by the number of results received.
The
cursor is a Unix timestamp. Treat it as opaque. Rare duplicates are possible across pages. Deduplicate by ID.
Useful jq Filters
# Business class availability sorted by cheapest miles ... | jq '[.data[] | select(.JAvailable == true) | {date: .Date, origin: .Route.OriginAirport, dest: .Route.DestinationAirport, miles: (.JMileageCost | tonumber), seats: .JRemainingSeats, airlines: .JAirlines, source: .Source, direct: .JDirect}] | sort_by(.miles)' # Saver vs dynamic detection: if miles < threshold, likely saver # AA: J saver ~57.5K for transatlantic, dynamic 100K+ # United: J saver ~60-80K for transatlantic, dynamic 100K+ ... | jq '[.data[] | select(.JAvailable == true) | {date: .Date, miles: (.JMileageCost | tonumber), source: .Source, type: (if (.JMileageCost | tonumber) < 80000 then "likely_saver" else "dynamic" end)}]' # Economy availability with 2+ seats ... | jq '[.data[] | select(.YAvailable == true and .YRemainingSeats >= 2) | {date: .Date, origin: .Route.OriginAirport, dest: .Route.DestinationAirport, miles: (.YMileageCost | tonumber), seats: .YRemainingSeats, source: .Source}] | sort_by(.miles)' # All available cabins for a date range ... | jq '[.data[] | {date: .Date, origin: .Route.OriginAirport, dest: .Route.DestinationAirport, source: .Source, economy: (if .YAvailable then .YMileageCost else null end), business: (if .JAvailable then .JMileageCost else null end), first: (if .FAvailable then .FMileageCost else null end)}]' # Direct flights only in business ... | jq '[.data[] | select(.JAvailable == true and .JDirect == true) | {date: .Date, miles: .JMileageCost, airlines: .JAirlines, source: .Source}]' # Check data freshness (stale = potential phantom) ... | jq '[.data[] | {date: .Date, source: .Source, last_seen: .ComputedLastSeen}]'
Workflow: Trip Planning Search
- Cached Search across multiple airports and date ranges to see what's available
- Compare programs: check which
has the cheapest miles for the same routesource - Identify saver vs dynamic: low prices (e.g., 57.5K J for AA transatlantic) = saver. High prices (100K+) = dynamic.
- Get Trip Details on promising availability IDs for flight times, connections, and booking links
- Verify freshness: check
. If stale (hours old), the availability may be phantom.ComputedLastSeen - Cross-reference: check the airline's own site or call to confirm before transferring points
- Use booking_links from trip response to book directly on each program's site
- Cross-reference with AwardWallet balances to confirm enough points
Why Results May Be Empty
- No saver space released. Most programs only show saver awards to partners. Check the airline's own site.
- Elite/status restrictions. Some seats only visible to elite members or cardholders.
- Route not monitored. If the route isn't in Get Routes, cached search won't have it. Live search can check any route.
- Search window limits. Airlines release awards at different advance windows. Check release dates tool.
- Already booked. Award space changes constantly. Set alerts for notification when it opens.
Specialized Tools (Web UI Only)
These tools are available on the seats.aero website but not via the Partner API:
- Alaska Upgrade Finder (seats.aero/alaska/upgrades): Find confirmable upgrade space for Alaska elites using Atmos Gold Guest Upgrade certificates. Dedicated tool for scanning upgrade availability.
- Etihad First Class Finder (seats.aero/etihad/first): Dedicated tracker for newly available Etihad First Class seats. Etihad F is rare and highly sought after. This tool surfaces new availability faster than general search.
- Price History: Fare trends over time are available on the web UI but not via API. Use the
skill (Patchright browser automation) to access price history charts.seats-aero-web
Alert Features
When creating alerts on seats.aero:
- Time filters: Departure and arrival time ranges are available as alert filters. Useful for avoiding red-eyes or ensuring morning arrivals.
- Cabin, route, and program filters: Standard alert parameters.
- Alerts notify via email or push when matching availability appears.
Third-Party Integrations
Pro account holders can connect seats.aero to third-party apps via the "Connect seats.aero" button on the website.
- aeroconnections.app (by @wavydavy): Visualization tool that shows available connections from specific airports. Log in with your Pro account to see current award space mapped visually. Useful for exploring routing options you wouldn't think to search.
Notes
- All times in responses are local airport times.
is in cents (divide by 100 for dollars).TotalTaxes
in availability objects is a string. In trip objects it's an integer.MileageCost
in trip objects shows what booking through an alliance partner would cost.AllianceCost- Dynamic pricing filters are on by default. Pass
to see everything.include_filtered=true - Availability data is cached, not live. Check
for freshness.ComputedLastSeen - Pro users: 1,000 API calls/day. Failed live searches don't count.
- Free users can only search 30 days ahead. Pro unlocks 60+ days.
- Post-search filters (max points, stops, duration, etc.) now apply directly to flight detail results, not just the search results list.