Connects Claude to OneBusAway transit APIs for real-time vehicle tracking and schedule lookups. You get 15 tools covering stop discovery by location or name, live arrival predictions with GPS positions and schedule deviations, full trip sequences, agency-wide vehicle positions, and static timetables. The arrivals endpoint distinguishes GPS-tracked predictions from schedule-only data and surfaces active service alerts inline. Useful for building transit assistants that need to answer "when's my bus" with actual vehicle locations, not just printed schedules. Works over stdio or streamable HTTP, with a public hosted instance at onebusaway.caseyjhand.com/mcp if you don't want to run it locally. Puget Sound is the reference region, but any OneBusAway deployment works with the right API key.
Query stops, routes, real-time arrivals, vehicle positions, and schedules from OneBusAway transit APIs via MCP. STDIO or Streamable HTTP.
15 tools covering the full OneBusAway transit data surface — discovery, real-time operations, schedules, alerts, and block schedules:
| Tool | Description |
|---|---|
onebusaway_list_agencies | List all transit agencies on this OneBusAway instance with IDs, contact info, and geographic coverage |
onebusaway_find_stops | Find bus stops near a lat/lon within a configurable radius, optionally filtered by stop code |
onebusaway_search_stops | Search stops by name or code string to resolve a human-readable name to a stop ID |
onebusaway_get_stop | Fetch details for a specific stop by agency-prefixed ID |
onebusaway_find_routes | Find transit routes near a lat/lon, optionally filtered by name or number |
onebusaway_search_routes | Search routes by name or number to resolve a route short name to a route ID |
onebusaway_get_route | Fetch details for a specific route by agency-prefixed ID |
onebusaway_list_routes_for_agency | List all routes operated by an agency |
onebusaway_get_arrivals | Real-time arrivals and departures at a stop — GPS-tracked predictions, schedule deviation, vehicle positions, and active alerts |
onebusaway_get_trip | Real-time status and full stop sequence for an active trip |
onebusaway_get_vehicles | Real-time positions of all active vehicles for an agency, optionally filtered to one route |
onebusaway_get_schedule_for_stop | Full-day departure schedule for a stop by route and direction |
onebusaway_get_schedule_for_route | Full-day schedule for a route — all trips and stop sequences |
onebusaway_get_alert | Fetch full service alert detail by situation ID — summary, description, reason, affected stops/routes, consequence, and active time windows |
onebusaway_get_block | Fetch the full-day block schedule for a vehicle by block ID — all trips in order with stop times, useful for fleet tracking |
onebusaway_find_stopsFind bus stops near a geographic location.
75403)limitExceeded flag signals when more stops exist beyond the returned setonebusaway_get_arrivalsonebusaway_get_arrivalsReal-time arrivals and departures at a stop.
minutesBefore, minutesAfter — defaults 5/35)predicted boolean distinguishes GPS-tracked estimates from schedule-only projectionstripId on each arrival feeds onebusaway_get_trip for vehicle trackingonebusaway_get_tripReal-time status and stop sequence for a specific trip.
in_progress, layover_before, layover_during)serviceDateMs for looking up trips from a prior service dayonebusaway_get_vehiclesReal-time positions of all active vehicles for an agency.
routeId filter (applied client-side after fetching all agency vehicles)predicted flag distinguish actively-reporting vehicles from stale entriesonebusaway_search_routesSearch for routes by name or number across the instance.
onebusaway_find_routes (lat/lon) or onebusaway_list_routes_for_agency are the alternativesonebusaway_get_schedule_for_stop and onebusaway_get_schedule_for_routeStatic schedule lookups — full-day timetables without real-time data.
onebusaway_get_schedule_for_stop: all departures grouped by route and direction, with trip IDs for follow-up callsonebusaway_get_schedule_for_route: all trips for the route with full stop sequences and GTFS stop times| Type | Name | Description |
|---|---|---|
| Resource | onebusaway://stop/{stopId} | Stop metadata — name, coordinates, served routes, and wheelchair accessibility |
| Resource | onebusaway://route/{routeId} | Route metadata — short name, description, agency, and schedule URL |
All resource data is also reachable via onebusaway_get_stop and onebusaway_get_route. Stop and route IDs use agency-prefixed format: {agencyId}_{localId} (e.g. 1_75403, 1_100259).
Built on @cyanheads/mcp-ts-core:
none, jwt, oauthin-memory, filesystem, Supabase, Cloudflare KV/R2/D1OneBusAway-specific:
onebusaway-sdk with typed error classification (NotFound, RateLimited, ServiceUnavailable)api.pugetsound.onebusaway.org) — works with ONEBUSAWAY_API_KEY=TEST for developmentONEBUSAWAY_BASE_URL for any OneBusAway-compatible instance (NYC, Washington DC, Tampa, etc.)Agent-friendly output:
predicted boolean on every arrival and vehicle distinguishes GPS-tracked data from schedule-only projections — agents branch on data, not string parsingtripId from arrivals feeds onebusaway_get_trip; stopId from searches feeds arrivals; agencyId from list feeds vehicles and route listingonebusaway_search_routes 404 → fallback to onebusaway_find_routes or onebusaway_list_routes_for_agency)Add the following to your MCP client configuration file. ONEBUSAWAY_API_KEY=TEST works on the Puget Sound instance without registration.
{
"mcpServers": {
"onebusaway-mcp-server": {
"type": "stdio",
"command": "bunx",
"args": ["@cyanheads/onebusaway-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info",
"ONEBUSAWAY_API_KEY": "TEST"
}
}
}
}
Or with npx (no Bun required):
{
"mcpServers": {
"onebusaway-mcp-server": {
"type": "stdio",
"command": "npx",
"args": ["-y", "@cyanheads/onebusaway-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info",
"ONEBUSAWAY_API_KEY": "TEST"
}
}
}
}
Or with Docker:
{
"mcpServers": {
"onebusaway-mcp-server": {
"type": "stdio",
"command": "docker",
"args": [
"run", "-i", "--rm",
"-e", "MCP_TRANSPORT_TYPE=stdio",
"-e", "ONEBUSAWAY_API_KEY=TEST",
"ghcr.io/cyanheads/onebusaway-mcp-server:latest"
]
}
}
}
For Streamable HTTP, set the transport and start the server:
MCP_TRANSPORT_TYPE=http MCP_HTTP_PORT=3010 ONEBUSAWAY_API_KEY=TEST bun run start:http
# Server listens at http://localhost:3010/mcp
TEST works on the Puget Sound instance for development. For production use or other instances, register at the relevant agency's developer portal.git clone https://github.com/cyanheads/onebusaway-mcp-server.git
cd onebusaway-mcp-server
bun install
cp .env.example .env
# edit .env — set ONEBUSAWAY_API_KEY if needed
All configuration is validated at startup via Zod schemas. Key environment variables:
| Variable | Description | Default |
|---|---|---|
ONEBUSAWAY_API_KEY | OneBusAway API key. TEST works on Puget Sound for development. | TEST |
ONEBUSAWAY_BASE_URL | Base URL for the OneBusAway instance. | https://api.pugetsound.onebusaway.org |
MCP_TRANSPORT_TYPE | Transport: stdio or http. | stdio |
MCP_HTTP_PORT | HTTP server port. | 3010 |
MCP_HTTP_ENDPOINT_PATH | HTTP endpoint path. | /mcp |
MCP_AUTH_MODE | Auth mode: none, jwt, or oauth. | none |
MCP_LOG_LEVEL | Log level (debug, info, notice, warning, error). | info |
LOGS_DIR | Directory for log files (Node.js only). | <project-root>/logs |
STORAGE_PROVIDER_TYPE | Storage backend: in-memory, filesystem, supabase, cloudflare-kv/r2/d1. | in-memory |
OTEL_ENABLED | Enable OpenTelemetry instrumentation. | false |
See .env.example for the full list of optional overrides.
Build and run the production version:
# One-time build
bun run rebuild
# Run the built server
bun run start:http
# or
bun run start:stdio
Run checks and tests:
bun run devcheck # Lint, format, typecheck, security, changelog sync
bun run test # Vitest test suite
bun run lint:mcp # Validate MCP definitions against spec
docker build -t onebusaway-mcp-server .
docker run --rm -e ONEBUSAWAY_API_KEY=TEST -p 3010:3010 onebusaway-mcp-server
The Dockerfile defaults to HTTP transport, stateless session mode, and logs to /var/log/onebusaway-mcp-server. OpenTelemetry peer dependencies are installed by default — build with --build-arg OTEL_ENABLED=false to omit them.
| Directory | Purpose |
|---|---|
src/index.ts | createApp() entry point — registers tools/resources and inits the OneBusAway service. |
src/config/server-config.ts | Server-specific env var parsing: ONEBUSAWAY_API_KEY, ONEBUSAWAY_BASE_URL. |
src/mcp-server/tools | Tool definitions (*.tool.ts). 13 tools across discovery, real-time, and schedule operations. |
src/mcp-server/resources | Resource definitions (*.resource.ts). Stop and route metadata resources. |
src/services/onebusaway | OneBusAway service — wraps onebusaway-sdk, typed error classification, domain types. |
tests/ | Unit and integration tests mirroring src/. 77 tests covering all tools and resources. |
See CLAUDE.md for development guidelines and architectural rules. The short version:
try/catch in tool logicctx.log for request-scoped logging, ctx.state for tenant-scoped storagecreateApp() arrays in src/index.tsIssues and pull requests are welcome. Run checks and tests before submitting:
bun run devcheck
bun run test
Transit data is sourced from the Puget Sound OneBusAway API, operated by Sound Transit and King County Metro. Use of this data is governed by the Sound Transit Transit Data Terms of Use.
Downstream users of this server's hosted endpoint receive data subject to those terms. Key obligations include:
Apache-2.0 — see LICENSE for details.
ONEBUSAWAY_API_KEYdefault: TESTOneBusAway API key. TEST works on the Puget Sound instance for development.
ONEBUSAWAY_BASE_URLdefault: https://api.pugetsound.onebusaway.orgBase URL for the OneBusAway instance. Defaults to Puget Sound.
MCP_LOG_LEVELdefault: infoSets the minimum log level for output (e.g., 'debug', 'info', 'warn').
MCP_HTTP_HOSTdefault: 127.0.0.1The hostname for the HTTP server.
MCP_HTTP_PORTdefault: 3010The port to run the HTTP server on.
MCP_HTTP_ENDPOINT_PATHdefault: /mcpThe endpoint path for the MCP server.
MCP_AUTH_MODEdefault: noneAuthentication mode to use: 'none', 'jwt', or 'oauth'.