2020-11-30 21:26:14 +00:00
|
|
|
const OnStar = require('onstarjs');
|
|
|
|
const mqtt = require('async-mqtt');
|
|
|
|
const uuidv4 = require('uuid').v4;
|
|
|
|
const _ = require('lodash');
|
|
|
|
const Vehicle = require('./vehicle');
|
2020-12-02 19:11:50 +00:00
|
|
|
const {Diagnostic} = require('./diagnostic');
|
|
|
|
const MQTT = require('./mqtt');
|
2020-11-30 21:26:14 +00:00
|
|
|
|
|
|
|
const onstarConfig = {
|
|
|
|
deviceId: process.env.ONSTAR_DEVICEID || uuidv4(),
|
|
|
|
vin: process.env.ONSTAR_VIN,
|
|
|
|
username: process.env.ONSTAR_USERNAME,
|
|
|
|
password: process.env.ONSTAR_PASSWORD,
|
|
|
|
onStarPin: process.env.ONSTAR_PIN,
|
2020-12-02 20:21:01 +00:00
|
|
|
checkRequestStatus: process.env.ONSTAR_SYNC === "true" || true,
|
2020-12-02 19:11:50 +00:00
|
|
|
refreshInterval: parseInt(process.env.ONSTAR_REFRESH) || (30 * 60 * 1000) // 30 min
|
2020-11-30 21:26:14 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const mqttConfig = {
|
2020-12-02 19:11:50 +00:00
|
|
|
host: process.env.MQTT_HOST || 'localhost',
|
2020-11-30 21:26:14 +00:00
|
|
|
username: process.env.MQTT_USERNAME,
|
|
|
|
password: process.env.MQTT_PASSWORD,
|
2020-12-02 19:11:50 +00:00
|
|
|
port: parseInt(process.env.MQTT_PORT) || 1883,
|
|
|
|
tls: process.env.MQTT_TLS || false,
|
|
|
|
prefix: process.env.MQTT_PREFIX || 'homeassistant',
|
2020-11-30 21:26:14 +00:00
|
|
|
};
|
|
|
|
|
2020-12-02 19:11:50 +00:00
|
|
|
let loop;
|
|
|
|
(async () => {
|
|
|
|
try {
|
|
|
|
const onStar = OnStar.create(onstarConfig);
|
|
|
|
console.log('Requesting vehicles.');
|
|
|
|
const vehiclesRes = await onStar.getAccountVehicles();
|
|
|
|
console.log(_.get(vehiclesRes, 'status'));
|
|
|
|
const vehicles = _.map(
|
|
|
|
_.get(vehiclesRes, 'response.data.vehicles.vehicle'),
|
|
|
|
v => new Vehicle(v)
|
|
|
|
);
|
|
|
|
console.log('Vehicles returned:');
|
|
|
|
for (const v of vehicles) {
|
|
|
|
console.log(v.toString());
|
|
|
|
}
|
2020-12-28 02:21:13 +00:00
|
|
|
|
2020-12-02 19:11:50 +00:00
|
|
|
const mqttHA = new MQTT('homeassistant', vehicles[0].vin);
|
2020-12-28 02:21:13 +00:00
|
|
|
const availTopic = mqttHA.getAvailabilityTopic();
|
|
|
|
const client = await mqtt.connectAsync(`${mqttConfig.tls
|
|
|
|
? 'mqtts' : 'mqtt'}://${mqttConfig.host}:${mqttConfig.port}`, {
|
|
|
|
username: mqttConfig.username,
|
|
|
|
password: mqttConfig.password,
|
|
|
|
will: {topic: availTopic, payload: 'false', retain: true}
|
|
|
|
});
|
|
|
|
|
|
|
|
await client.publish(availTopic, 'true', {retain: true});
|
2020-11-30 21:26:14 +00:00
|
|
|
|
2020-12-28 02:21:13 +00:00
|
|
|
const configurations = new Map();
|
2020-12-02 19:11:50 +00:00
|
|
|
const run = async () => {
|
2020-12-28 02:21:13 +00:00
|
|
|
const states = new Map();
|
2020-12-02 19:11:50 +00:00
|
|
|
// Note: the library is set to use only the configured VIN, but using multiple for future proofing.
|
|
|
|
for (const v of vehicles) {
|
|
|
|
console.log('Requesting diagnostics:')
|
|
|
|
const statsRes = await onStar.diagnostics({
|
|
|
|
diagnosticItem: v.getSupported()
|
|
|
|
});
|
|
|
|
console.log(_.get(statsRes, 'status'));
|
|
|
|
const stats = _.map(
|
|
|
|
_.get(statsRes, 'response.data.commandResponse.body.diagnosticResponse'),
|
|
|
|
d => new Diagnostic(d)
|
|
|
|
);
|
2020-11-30 21:26:14 +00:00
|
|
|
|
2020-12-02 19:11:50 +00:00
|
|
|
for (const s of stats) {
|
|
|
|
if (!s.hasElements()) {
|
|
|
|
continue;
|
|
|
|
}
|
2020-12-28 02:21:13 +00:00
|
|
|
// configure once, then set or update states
|
2020-12-02 19:11:50 +00:00
|
|
|
for (const d of s.diagnosticElements) {
|
2020-12-28 02:21:13 +00:00
|
|
|
const topic = mqttHA.getConfigTopic(d)
|
|
|
|
const payload = mqttHA.getConfigPayload(s, d);
|
|
|
|
configurations.set(topic, {configured: false, payload});
|
2020-12-02 19:11:50 +00:00
|
|
|
}
|
2020-12-28 02:21:13 +00:00
|
|
|
|
|
|
|
const topic = mqttHA.getStateTopic(s);
|
|
|
|
const payload = mqttHA.getStatePayload(s);
|
|
|
|
states.set(topic, payload);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (let [topic, config] of configurations) {
|
|
|
|
// configure once
|
|
|
|
if (!config.configured) {
|
|
|
|
config.configured = true;
|
|
|
|
const {payload} = config;
|
|
|
|
console.log(`${topic} ${JSON.stringify(payload)}`);
|
|
|
|
await client.publish(topic, JSON.stringify(payload), {retain: true});
|
2020-12-02 19:11:50 +00:00
|
|
|
}
|
|
|
|
}
|
2020-12-28 02:21:13 +00:00
|
|
|
// update states
|
|
|
|
for (let [topic, state] of states) {
|
|
|
|
console.log(`${topic} ${JSON.stringify(state)}`);
|
|
|
|
await client.publish(topic, JSON.stringify(state), {retain: true});
|
|
|
|
}
|
2020-12-02 19:11:50 +00:00
|
|
|
};
|
2020-11-30 21:26:14 +00:00
|
|
|
|
2020-12-02 19:11:50 +00:00
|
|
|
const main = () => run()
|
|
|
|
.then(() => console.log('Done, sleeping.'))
|
|
|
|
.catch(e => console.error(e))
|
2020-11-30 21:26:14 +00:00
|
|
|
|
2020-12-28 02:21:13 +00:00
|
|
|
await main();
|
2020-12-02 19:11:50 +00:00
|
|
|
loop = setInterval(main, onstarConfig.refreshInterval);
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
2020-11-30 21:26:14 +00:00
|
|
|
}
|
2020-12-02 19:11:50 +00:00
|
|
|
})();
|