-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathcompute-sorted-connections.js
86 lines (71 loc) · 2.63 KB
/
compute-sorted-connections.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
'use strict'
const debug = require('debug')('gtfs-utils:compute-sorted-connections')
const {gte} = require('sorted-array-functions')
const inMemoryStore = require('./lib/in-memory-store')
const readStopTimezones = require('./lib/read-stop-timezones')
const readTrips = require('./read-trips')
const readServicesAndExceptions = require('./read-services-and-exceptions')
const computeConnections = require('./compute-connections')
const resolveTime = require('./lib/resolve-time')
const computeSortedConnections = async (readFile, timezone, filters = {}, opt = {}) => {
if ('string' !== typeof timezone || !timezone) {
throw new Error('timezone must be a non-empty string.')
}
filters = {
stop: () => true,
...filters
}
const {
createStore,
} = {
createStore: inMemoryStore,
...opt,
}
debug('reading stops.stop_timezone')
// stop.stop_id -> stop.stop_timezone || parent.stop_timezone
const stopTimezones = await readStopTimezones(readFile, filters, createStore)
debug('reading trips')
const svcIdsRouteIdsByTrip = await readTrips(readFile, filters, {
...opt,
formatTrip: t => [t.service_id, t.route_id],
})
debug('reading services & exceptions')
const _services = readServicesAndExceptions(readFile, timezone, filters)
const services = createStore() // by service ID
for await (const [id, dates] of _services) {
await services.set(id, dates)
}
// todo: use store API to support memory-constrained environments
const sortedConnections = []
const compareConnections = (a, b) => a.departure - b.departure
debug('reading connections')
const connectionsByTrip = computeConnections(readFile, filters, opt)
for await (const connections of connectionsByTrip) {
if (connections.length === 0) continue
const _ = await svcIdsRouteIdsByTrip.get(connections[0].tripId)
if (!_) continue
const [serviceId, routeId] = _
const dates = await services.get(serviceId)
if (!dates) continue // todo: log error?
for (const c of connections) {
for (let i = 0; i < dates.length; i++) {
const fromTz = (await stopTimezones.get(c.fromStop)) || timezone
const toTz = (await stopTimezones.get(c.toStop)) || timezone
const newCon = {
tripId: c.tripId,
serviceId, routeId,
fromStop: c.fromStop,
departure: resolveTime(fromTz, dates[i], c.departure),
toStop: c.toStop,
arrival: resolveTime(toTz, dates[i], c.arrival),
headwayBased: !!c.headwayBased,
}
const idx = gte(sortedConnections, newCon, compareConnections)
if (idx === -1) sortedConnections.push(newCon)
else sortedConnections.splice(idx, 0, newCon)
}
}
}
return sortedConnections
}
module.exports = computeSortedConnections