The open HFP API can be used to subscribe to vehicle movements in soft real time.
Most of the vehicles in the HSL area should publish their status, including their position, once per second. The devices of the end users, e.g. smartphones, may subscribe to receive the relevant messages based on their context, e.g. filtered on the mode of transport, the route ID, the geographical region etc. The subscription scope is specified by the MQTT topic structure of the API.
Vehicle positions are also available in GTFS-RT format, see HSL GTFS-RT API documentation
Try this example to get an quick idea of what kind of data is available from the API:
npm install -g mqtt
mqtt subscribe -h mqtt.hsl.fi -p 8883 -l mqtts -v -t "/hfp/v2/journey/#"
URL | Description |
---|---|
mqtts://mqtt.hsl.fi:8883/ |
The bare MQTT protocol with TLS, lightweight. |
wss://mqtt.hsl.fi:443/ |
MQTT over WebSockets with TLS, for browsers. |
mqtt://mqtt.hsl.fi:1883/ |
The bare MQTT protocol without TLS, lightweight. Prefer the port 8883 to respect the locational privacy of your users. |
ws://mqtt.hsl.fi:1883/ |
MQTT over WebSockets without TLS. Prefer the port 443 to respect the locational privacy of your users. |
From a client's point of view, every MQTT message has two parts, the topic and the binary payload.
The topic looks similar to a file path with levels separated by /
.
When subscribing, all values for a level or several levels may be specified with a wildcard.
+
refers to all values for one level, #
refers to all values of the parent level and its child levels.
See the advanced examples further below on how to tailor the subscription to your needs.
Here is an example of an HFP message topic:
/hfp/v2/journey/ongoing/vp/bus/0055/01216/1069/1/Malmi/07:20/1130106/2/60;24/19/73/44
It can be split into these parts:
/<prefix>/<version>/<journey_type>/<temporal_type>/<event_type>/<transport_mode>/<operator_id>/<vehicle_number>/<route_id>/<direction_id>/<headsign>/<start_time>/<next_stop>/<geohash_level>/<geohash>/<sid>/#
Topic levels up to vehicle_number
are present in all types of topics and levels after vehicle_number
are only present with vehicle journey position messages (i.e. when journey_type
is journey
).
Note: For future proofing your subscription, add wildcard #
to the end of the topic. We may change the HFP topic structure by adding new filters to the end without updating the version number.
Attribute | Description |
---|---|
prefix |
/hfp/ is the root of the topic tree. |
version |
v2 is the current version of the HFP topic and the payload format. |
journey_type |
The type of the journey. Either journey , deadrun or signoff .journey refers to a vehicle that is running on a specific public transport journey.deadrun refers to a vehicle that is not on any specific route, but instead coming from a depot, for example.signoff is used when the vehicle PC is shut down.Note: deadrun and signoff messages are only available for authorized users. |
temporal_type |
The status of the journey, ongoing or upcoming .ongoing describes a journey that is currently in operation.upcoming refers to the next expected journey of the same vehicle. upcoming messages are broadcasted shortly before the start of the next journey. One use of upcoming is to show the relevant vehicle to your users even before the driver has signed on to the journey that your users are interested in. |
event_type |
One of vp , due , arr , dep , ars , pde , pas , wait , doo , doc , tlr , tla , da , dout , ba , bout , vja , vjout .See descriptions for these values below. |
transport_mode |
The type of the vehicle. One of bus , tram , train , ferry , metro , ubus (used by U-line buses and other vehicles with limited realtime information) or robot (used by robot buses). |
operator_id |
The unique ID of the operator that owns the vehicle. See the list of operators below. Note: Operator ids must be exactly 4 digits long in the topic filter, so prefix them with zeroes if needed (e.g. 80 → 0080 ) |
vehicle_number |
The vehicle number that can be seen painted on the side of the vehicle, often next to the front door. Different operators may use overlapping vehicle numbers. operator_id/vehicle_number uniquely identifies the vehicle.Note: Vehicle numbers must be exactly 5 digits long in the topic filter, so prefix them with zeroes if needed. |
route_id |
The ID of the route the vehicle is running on. This matches route_id in GTFS (field gtfsId of Route in the routing API). |
direction_id |
The line direction of the trip, either 1 or 2 .Note: This does not exactly match direction_id in GTFS or the routing API.Value 1 here is same as 0 in GTFS and the Routing API.Value 2 here is same as 1 in GTFS and the Routing API. |
headsign |
The destination name, e.g. Aviapolis . Note: This does NOT match trip_headsign in GTFS exactly. |
start_time |
The scheduled start time of the trip, i.e. the scheduled departure time from the first stop of the trip. The format follows HH:mm in 24-hour local time, not the 30-hour overlapping operating days present in GTFS. |
next_stop |
The ID of next stop or station. Updated on each departure from or passing of a stop. EOL (end of line) after final stop and empty if the vehicle is leaving HSL area. Matches stop_id in GTFS (value of gtfsId field, without HSL: prefix, in Stop type in the routing API). |
geohash_level |
The geohash level represents the magnitude of change in the GPS coordinates since the previous message from the same vehicle. More exactly, geohash_level is equal to the minimum of the digit positions of the most significant changed digit in the latitude and the longitude since the previous message. For example, if the previous message has value (60.12345, 25.12345) for (lat , long ) and the current message has value (60.12499, 25.12388), then the third digit of the fractional part is the most significant changed digit and geohash_level has value 3 .However, geohash_level value 0 is overloaded. geohash_level is 0 if:
0 the client gets the most important status changes. The rough percentages of messages with a specific geohash_level value out of all ongoing messages are:
|
geohash |
The latitude and the longitude of the vehicle. The digits of the integer parts are separated into their own level in the format <lat>;<long> , e.g. 60;24 . The digits of the fractional parts are split and interleaved into a custom format so that e.g. (60.123, 24.789) becomes 60;24/17/28/39 . This format enables subscribing to specific geographic boundaries easily.If the coordinates are missing, geohash_level and geohash have the concatenated value 0//// .This geohash scheme is greatly simplified from the original geohash scheme |
sid |
Junction ID, corresponds to sid in the payload. Only available for tlr and tla event types, empty for other event types. |
The most notable change in HFP 2.0 is introduction of different types of messages. Whereas HFP 1.0 had only vehicle position messages, in HFP 2.0 vehicles also send messages from different types of event happening during the journey (for example, arriving to a stop etc.).
Note: events are not available for metros (metro
), U-line buses (ubus
), robot buses (robot
) and ferries (ferry
).
List of possible events:
Event type | Description |
---|---|
vp |
Vehicle position |
due |
Vehicle will soon arrive to a stop |
arr |
Vehicle arrives inside of a stop radius |
dep |
Vehicle departs from a stop and leaves the stop radius |
ars |
Vehicle has arrived to a stop |
pde |
Vehicle is ready to depart from a stop |
pas |
Vehicle passes through a stop without stopping |
wait |
Vehicle is waiting at a stop |
doo |
Doors of the vehicle are opened |
doc |
Doors of the vehicle are closed |
tlr |
Vehicle is requesting traffic light priority |
tla |
Vehicle receives a response to traffic light priority request |
da |
Driver signs in to the vehicle |
dout |
Driver signs out of the vehicle |
ba |
Driver selects the block that the vehicle will run |
bout |
Driver signs out from the selected block (usually from a depot) |
vja |
Vehicle signs in to a service journey (i.e. a single public transport journey from location A to location B, also known as trip) |
vjout |
Vehicle signs off from a service journey, after reaching the final stop |
The payload is an UTF-8 encoded, compact JSON string. The JSON contains a single key-value pair where the key is equivalent to the event type of the message and the value is an object describing the event. Fields of the event object depend on event type.
Here is an example of a vehicle position message (with VP
key):
{
"VP": {
"desi": "551",
"dir": "1",
"oper": 12,
"veh": 10,
"tst": "2019-05-23T14:15:16.000Z",
"tsi": 1416308975,
"spd": 12.5,
"hdg": 354,
"lat": 24.9435,
"long": 60.1967,
"acc": -12.34,
"dl": 600,
"odo": 45.12,
"drst": 0,
"oday": "2019-05-23",
"jrn": 1,
"line": 264,
"start": "13:40",
"loc": "GPS",
"stop": null,
"route": "2551",
"occu": 0
}
}
The fields are described below:
Field | Type | Not available on events | Description |
---|---|---|---|
desi |
String | da , dout , ba , bout |
Route number visible to passengers. |
dir |
String | da , dout , ba , bout |
Route direction of the trip. After type conversion matches direction_id in GTFS and the topic. Either "1" or "2" . |
oper |
Integer | Unique ID of the operator running the trip (i.e. this value can be different than the operator ID in the topic, for example if the service has been subcontracted to another operator). The unique ID does not have prefix zeroes here. |
|
veh |
Integer | Vehicle number that can be seen painted on the side of the vehicle, often next to the front door. Different operators may use overlapping vehicle numbers. Matches vehicle_number in the topic except without the prefix zeroes. |
|
tst |
String | UTC timestamp with millisecond precision from the vehicle in ISO 8601 format (yyyy-MM-dd'T'HH:mm:ss.SSSZ ). |
|
tsi |
Integer | Unix time in seconds from the vehicle. | |
spd |
Floating-point number | Speed of the vehicle, in meters per second (m/s). | |
hdg |
Integer | Heading of the vehicle, in degrees (⁰) starting clockwise from geographic north. Valid values are on the closed interval [0, 360]. | |
lat |
Floating-point number | WGS 84 latitude in degrees.null if location is unavailable. |
|
long |
Floating-point number | WGS 84 longitude in degrees.null if location is unavailable. |
|
acc |
Floating-point number | Acceleration (m/s^2), calculated from the speed on this and the previous message. Negative values indicate that the speed of the vehicle is decreasing. | |
dl |
Integer | da , dout , ba , bout |
Offset from the scheduled timetable in seconds (s). Negative values indicate lagging behind the schedule, positive values running ahead of schedule. |
odo |
Integer | The odometer reading in meters (m) since the start of the trip. Currently the values not very reliable. | |
drst |
Integer | Door status.0 if all the doors are closed.1 if any of the doors are open. |
|
oday |
String | da , dout |
Operating day of the trip. The exact time when an operating day ends depends on the route. For most routes, the operating day ends at 4:30 AM on the next day. In that case, for example, the final moment of the operating day "2018-04-05" would be at 2018-04-06T04:30 local time. |
jrn |
Integer | da , dout , ba , bout |
Internal journey descriptor, not meant to be useful for external use. |
line |
Integer | da , dout , ba , bout |
Internal line descriptor, not meant to be useful for external use. |
start |
String | da , dout , ba , bout |
Scheduled start time of the trip, i.e. the scheduled departure time from the first stop of the trip. The format follows HH:mm in 24-hour local time, not the 30-hour overlapping operating days present in GTFS. Matches start_time in the topic. |
loc |
String | Location source, either GPS , ODO , MAN , DR or N/A .
|
|
stop |
String | da , dout , ba , bout |
ID of the stop related to the event (e.g. ID of the stop where the vehicle departed from in case of dep event or the stop where the vehicle currently is in case of vp event).null if the event is not related to any stop. |
route |
String | da , dout , ba , bout |
ID of the route the vehicle is currently running on. Matches route_id in the topic. |
occu |
Integer | da , dout , ba , bout |
Integer describing passenger occupancy level of the vehicle. Valid values are on interval [0, 100] .Currently passenger occupancy level is only available for Suomenlinna ferries as a proof-of-concept. The value will be available shortly after departure when the ferry operator has registered passenger count for the journey. For other vehicles, currently only values used are 0 (= vehicle has space and is accepting passengers) and 100 (= vehicle is full and might not accept passengers) |
seq |
Integer | Sequence number of the unit when the journey is operated with a vehicle that consists of multiple units (e.g. metros, trains). Sequence number starts from 1. Note: seq is currently only available for metros. |
|
label |
String | User visible label that helps to identify the vehicle. Currently available only for Suomenlinna ferries with values being vessel names. | |
ttarr |
String | vp , da , dout , ba , bout , vja , vjout |
UTC timestamp of scheduled arrival time to the stop |
ttdep |
String | vp , da , dout , ba , bout , vja , vjout |
UTC timestamp of scheduled departure time from the stop |
dr-type |
Integer | Other than da , dout , ba , bout , vja , vjout |
Type of the driver, either 0 or 1 .
|
tlp-requestid |
Integer | Other than tlr , tla |
Traffic light priority request ID. Valid values are on interval [0, 255] . |
tlp-requesttype |
String | Other than tlr |
Traffic light priority request type, either NORMAL , DOOR_CLOSE , DOOR_OPEN or ADVANCE . |
tlp-prioritylevel |
String | Other than tlr |
Priority level of a traffic light priority request. Either normal , high or norequest . |
tlp-reason |
String | Other than tlr |
Reason for not sending a traffic light priority request. Either GLOBAL , AHEAD , LINE or PRIOEXEP . |
tlp-att-seq |
Integer | Other than tlr |
Traffic light priority request attempt sequence number |
tlp-decision |
String | Other than tla |
Response for traffic light priority request. Either ACK or NAK . |
Also the following fields are available on traffic light priority request events (event type tlr
), but they probably have no use for 3rd party users:
Field | Type | Description |
---|---|---|
sid |
Integer | Junction ID |
signal-groupid |
Integer | Signal group (a group of traffic lights at a junction) ID |
tlp-signalgroupnbr |
Integer | ID of the specific traffic light in a signal group. Possibly negative |
tlp-line-configid |
Integer | ID of the line configuration in DOI |
tlp-point-configid |
Integer | Point configuration ID |
tlp-frequency |
Integer | Frequency used for traffic light prority request |
tlp-protocol |
String | Protocol used for traffic light priority request. Either MQTT or KAR-MQTT |
The numerical values for the different transit operators are listed below:
oper |
Operator name |
---|---|
6 |
Oy Pohjolan Liikenne Ab2 |
12 |
Helsingin Bussiliikenne Oy |
17 |
Tammelundin Liikenne Oy |
18 |
Oy Pohjolan Liikenne Ab |
20 |
Bus Travel Åbergin Linja Oy |
21 |
Bus Travel Oy Reissu Ruoti |
22 |
Nobina Finland Oy |
30 |
Savonlinja Oy |
36 |
Nurmijärven Linja Oy |
40 |
HKL-Raitioliikenne |
47 |
Taksikuljetus Oy |
50 |
HKL-Metroliikenne |
51 |
Korsisaari Oy |
54 |
V-S Bussipalvelut Oy |
58 |
Koillisen Liikennepalvelut Oy |
60 |
Suomenlinnan Liikenne Oy |
59 |
Tilausliikenne Nikkanen Oy |
89 |
Metropolia |
90 |
VR Oy |
130 |
Matkahuolto1 |
195 |
Siuntio1 |
1Multiple smaller operators operate under this operator ID
2This value is present only in the oper
field of the payload. Operator 6
is the same as 18
.
The HFP topic format forms a tree.
By combining wildcards with several topic filters in one MQTT SUBSCRIBE
packet you can carve quite interesting subsets of the tree to serve different use cases.
As the MQTT broker handles the resolving of the topic filters, handling messages for complicated subscriptions does not have inherent overhead in the client compared to simple subscriptions.
Go hog wild.
Below are sample subscriptions utilizing MQTT.js command line tools.
MQTT.js command line tools can be installed with:
npm install -g mqtt
If you insist on using mosquitto, try this for TLS access:
mosquitto_sub --capath "/etc/ssl/certs/" -h mqtt.hsl.fi -p 8883 -v -t "/hfp/v2/journey/#"
To get just the most significant vehicle position updates, use:
mqtt subscribe -h mqtt.hsl.fi -l mqtts -p 8883 -v \
-t "/hfp/v2/journey/ongoing/vp/+/+/+/+/+/+/+/+/0/#"
To subscribe to all vehicle position messages from vehicles currently on the route 551 (route_short_name
in GTFS) going in direction 1, subscribe to the corresponding route_id
2551:
mqtt subscribe -h mqtt.hsl.fi -l mqtts -p 8883 -v \
-t "/hfp/v2/journey/ongoing/vp/+/+/+/2551/1/#"
Subscribe to all vehicle position messages from trams with:
mqtt subscribe -h mqtt.hsl.fi -l mqtts -p 8883 -v \
-t "/hfp/v2/journey/ongoing/vp/tram/#"
Subscribe to vehicle position messages of a certain trip, even slightly before the driver has signed onto the trip:
mqtt subscribe -h mqtt.hsl.fi -l mqtts -p 8883 -v \
-t "/hfp/v2/journey/+/vp/+/+/+/1069/1/+/07:20/#"
Or if your users would find it confusing to see a vehicle going in the wrong direction, subscribe to the ongoing
messages only:
mqtt subscribe -h mqtt.hsl.fi -l mqtts -p 8883 -v \
-t "/hfp/v2/journey/ongoing/vp/+/+/+/1069/1/+/07:20/#"
To subscribe to all vehicles arriving to a certain stop, use:
mqtt subscribe -h mqtt.hsl.fi -l mqtts -p 8883 -v \
-t "/hfp/v2/journey/ongoing/arr/+/+/+/+/+/+/+/1293140/#"
Note: depending on the current traffic situation, you might have to wait a while to receive data
To subscribe to door status of a certain vehicle, use:
mqtt subscribe -h mqtt.hsl.fi -l mqtts -p 8883 -v \
-t "/hfp/v2/journey/ongoing/doo/+/0012/01312/#" \
-t "/hfp/v2/journey/ongoing/doc/+/0012/01312/#"
Note: you will have to change operator ID and vehicle number in the topic if vehicle 0012/01312
is not running currently.
Let's assume that you wish to subscribe to all action inside the following GeoJSON Polygon:
{
"type": "Feature",
"geometry": {
"type": "Polygon",
"coordinates": [
[
[24.9578905105, 60.1836538254],
[24.9646711349, 60.1836538254],
[24.9646711349, 60.1894146967],
[24.9578905105, 60.1894146967],
[24.9578905105, 60.1836538254]
]
]
}
}
The box bounded by the latitude interval [60.18, 60.19[ and the longitude interval [24.95, 24.97[ corresponds with the following HFP subscription:
mqtt subscribe -h mqtt.hsl.fi -l mqtts -p 8883 -v \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/#"
For the precision of one more digit of latitude and longitude, one would need 56 topic filters for the bounding box:
mqtt subscribe -h mqtt.hsl.fi -l mqtts -p 8883 -v \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/37/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/38/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/39/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/30/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/31/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/32/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/33/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/34/#" \
\
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/47/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/48/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/49/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/40/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/41/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/42/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/43/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/44/#" \
\
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/57/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/58/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/59/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/50/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/51/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/52/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/53/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/54/#" \
\
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/67/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/68/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/69/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/60/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/61/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/62/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/63/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/64/#" \
\
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/77/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/78/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/79/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/70/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/71/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/72/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/73/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/74/#" \
\
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/87/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/88/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/89/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/80/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/81/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/82/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/83/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/84/#" \
\
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/97/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/98/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/85/99/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/90/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/91/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/92/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/93/#" \
-t "/hfp/v2/journey/ongoing/+/+/+/+/+/+/+/+/+/+/60;24/19/86/94/#"
There is no need to restrict yourself to just one rectangle like above, though.
For example, you could try to generate an HFP subscription for all ongoing
vehicles in the minimal geographic area encompassing the Kontula borough with the precision of two digits in the fractional part.
The Routing API can be used to query a trip correspoding to a vehicle position message.
See this example on how to check if a vehicle is wheelchair accessible.