-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcontroller.js
105 lines (89 loc) · 3.31 KB
/
controller.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
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
import log from 'loglevel';
import { appConfig } from './config.js'
import { route2segments } from './route.js'
import { Vehicle } from './vehicle.js';
import mqtt from 'mqtt';
// vehicle controller
const vc = {
mqttClient: null,
vehicles: {},
routesMap: {},
// initialize everything based on configuration
init: function () {
// init all routes
for (const route of appConfig.routes) {
vc.routesMap[route.id] = route
route2segments(route)
}
// init vehicles
for (const vehConfig of appConfig.vehicles) {
const segments = vc.routesMap[vehConfig.route].segments
log.info(`Init ${vehConfig.number} ${vehConfig.type} vehicles on route ${vehConfig.route}`)
const numberOfVehicles = vehConfig.fleet ? 1 : vehConfig.number
for (let i = 1; i <= numberOfVehicles; i++) {
const vehicle = new Vehicle(i, segments, vehConfig)
vc.vehicles[vehicle.vehID] = vehicle
}
}
},
initMqtt: function (onConnected) {
// connect to broker
let startTime = new Date()
log.info(startTime.toLocaleString() + ": start to connect to " + appConfig.mqtt.brokerUrl)
vc.mqttClient = mqtt.connect(appConfig.mqtt.brokerUrl, appConfig.mqtt.clientOptions)
vc.mqttClient.on('connect', function () {
log.info("Connected, ready to send mqtt messages ...")
onConnected()
})
vc.mqttClient.on('message', function (topic, message) {
console.warn("Received message: " + message.toString())
})
vc.mqttClient.on('error', (err) => {
let nowTime = new Date()
log.error(nowTime.toLocaleString() + ": " + err.message)
if (nowTime - startTime > 300 * 1000) { // 5 minutes
log.error("TIME OUT, unable to connect to the broker " + appConfig.mqtt.brokerUrl)
vc.mqttClient.end()
process.exit()
}
})
},
start: function () {
vc.initMqtt(() => vc.startAllVehicles())
},
// start all vehicles
startAllVehicles: function () {
log.info(`Start all ${Object.keys(vc.vehicles).length} vehicles`)
log.info("Press Ctrl+C to exit ...")
for (const [vehId, vehicle] of Object.entries(vc.vehicles)) {
setTimeout(() => {
vehicle.move()
setInterval(() => vehicle.move(), 1000 * vehicle.reportInterval)
}, Math.random() * 1000 * vehicle.reportInterval)
}
},
onVehicleReport: function (payload) {
const topic = `${appConfig.topicPrefix}${payload.route}/${payload.vehType}/${payload.vehID}/` +
`${formatGpsCoord(payload.lat, "lat")}/${formatGpsCoord(payload.lng, "lng")}/` +
`${payload.heading.toFixed(0)}/${payload.status}`
vc.mqttClient.publish(topic, JSON.stringify(payload))
log.debug(topic)
}
}
// 5 decimal places should be good enough
// according to (https://sites.google.com/site/trescopter/Home/concepts/required-precision-for-gps-calculations)
const gpsSmallestUnit = "0.00001"
const fixedLength = gpsSmallestUnit.split(".")[1].length // 5
function formatGpsCoord(coord, dim) {
// lng: -180~180, lat: -90~90
let wholeLength = dim === "lng" ? fixedLength + 4 : fixedLength + 3
let result = coord.toFixed(fixedLength);
if (result[0] === '-') {
result = result.slice(1).padStart(wholeLength, '0');
result = '-' + result;
} else {
result = result.padStart(wholeLength, '0');
}
return result
}
export { vc as vehicleController }