TypeScript client for MTA realtime feeds and the hosted MTA API.
Use an API key from mtaapi.dev for route-aware static lookups and managed
realtime endpoints:
import { MTA } from "mta-js";
const mta = new MTA({
apiKey: process.env.MTA_API_KEY,
});
const nearby = await mta.stops.near({
lat: 40.7356,
lon: -73.9804,
modes: ["bus"],
route: "M23",
includeRoutes: true,
});
const lTrain = await mta.subway.arrivals({
stopId: "L08",
route: "L",
});Arrival rows may include display-oriented fields. destination depends on
destination metadata, while displayDirection may still be present as a
generic fallback label:
for (const arrival of lTrain) {
const direction =
arrival.displayDirection ??
(arrival.destination
? `toward ${arrival.destination}`
: arrival.headsign
? `toward ${arrival.headsign}`
: arrival.direction);
console.log(`${arrival.route.shortName} ${direction} from ${arrival.stop.name}`);
}NYC Subway realtime feeds use NYCT's north/south stop directions, even on
east-west lines. For the L train, mta-js accepts rider-facing east/west
aliases and maps them to the underlying feed directions.
When apiKey is present, mta-js sends requests to the hosted API at
https://www.mtaapi.dev by default. Override apiBaseUrl for tests or private
deployments.
You can still call MTA realtime feeds directly without the hosted API:
const mta = new MTA({
busTimeKey: process.env.MTA_BUS_KEY,
});
const buses = await mta.bus.arrivals({
stopId: "308214",
route: "M23",
});Direct feed mode has no bundled SQLite, Turso, or persistent GTFS database. If
you need richer local metadata, pass a small in-memory staticData seed:
const mta = new MTA({
staticData: {
stops: [
{
stop_id: "L08",
stop_name: "Bedford Av",
stop_lat: 40.717304,
stop_lon: -73.956872,
},
],
routes: [
{
route_id: "L",
route_short_name: "L",
route_long_name: "14 St-Canarsie Local",
},
],
},
staticDataMode: "subway",
});For production static stop search, prefer the hosted API. It serves a compact Blob-backed snapshot instead of requiring each SDK consumer to manage GTFS imports.
Route and stop inputs include generated autocomplete for known MTA values while
remaining permissive for future route and stop additions. Refresh the generated
types from the hosted stops snapshot by setting MTA_STOPS_SNAPSHOT_URL or
NEXT_PUBLIC_MTA_STOPS_SNAPSHOT_URL and running bun run generate:types. Bus
route autocomplete is generated from public MTA borough GTFS route files by
default; set MTA_BUS_GTFS_URLS to a comma-separated list of GTFS zip URLs to
override them.
mta.subway.arrivals(...)mta.bus.arrivals(...)mta.bus.vehicles(...)mta.alerts.current(...)mta.stops.near(...)