-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathindex.js
120 lines (107 loc) · 3.87 KB
/
index.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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
var mqtt = require('mqtt');
var spawn = require('child_process').spawn;
var async = require('async');
var fs = require('fs');
var config = {};
var cecClient = null;
var mqttClient = null;
var cmdQueue = {power: []};
var lastState = {power: {}};
var lastPowerCmdTime = {};
var startTime = new Date();
function reqPowerStatus(id, callback) {
cmdQueue.power.push(id);
cecClient.stdin.write('pow ' + id + '\n', callback);
}
async.series([
function readConfig(callback) {
fs.readFile('config.json', function(err, data) {
if(err) {
console.error('Failed to load config.json\nPlease ensure the file exists and is valid JSON.');
process.exit(1);
}
config = JSON.parse(data);
callback();
});
},
function connectMqtt(callback) {
// Connect MQTT client
mqttClient = mqtt.connect(config.mqttUrl);
console.log('Attempting to connect to MQTT server');
mqttClient.on('connect', function() {
for(var id in config.idMap) {
for(var subTopic of config.subTopics) {
console.log('Subscribing to ' + config.idMap[id].rootTopic + '/' + subTopic);
mqttClient.subscribe(config.idMap[id].rootTopic + '/' + subTopic);
}
}
callback();
});
},
function setupCecClient(callback) {
cecClient = spawn(config.cecClientCommand, config.cecClientArgs);
cecClient.stdout.on('data', function(data) {
for(var line of data.toString().split('\n')) {
if(line.length > 3 && line.indexOf('power status: ') === -1) console.log('cecClient> ' + line);
// Power status update
if(config.subTopics.indexOf('power') !== -1 && line.indexOf('power status: ') === 0) {
var id = cmdQueue.power.shift(); // Get the first element from the array
var reportedState = line.split(': ')[1];
var state = 'OFF';
switch(reportedState) {
case 'unknown':
case 'standby':
state = 'OFF';
break;
case 'in transition from standby to on':
case 'on':
state = 'ON';
break;
}
if(lastState.power[id] !== state && new Date() - (lastPowerCmdTime[id] || 0) > config.powerCmdWaitMs) {
console.log('publishing new state for ' + config.idMap[id].rootTopic + '/power : ' + state);
mqttClient.publish(config.idMap[id].rootTopic + '/power', state, {retain: true});
}
lastState.power[id] = state;
}
}
});
callback();
},
function setupMqttListeners(callback) {
mqttClient.on('message', function(topic, message) {
console.log('mqtt> ' + topic + ' : ' + message.toString());
if((new Date()) - startTime < 5000) {
console.log(' Assuming MQTT update is from initial subscription, ignoring');
return;
}
var id = 0;
for(id in config.idMap) { if(topic.indexOf(config.idMap[id].rootTopic) === 0) break; }
var subTopic = topic.split('/')[topic.split('/').length - 1];
switch(subTopic) {
case 'power':
if((message.toString() === 'ON' || message.toString() === 'OFF') && lastState.power[id] !== message.toString()) {
console.log('turning ' + id + ' ' + message.toString());
lastPowerCmdTime[id] = new Date();
switch(message.toString()) {
case 'OFF':
cecClient.stdin.write('standby ' + id + '\n');
break;
case 'ON':
cecClient.stdin.write('on ' + id + '\n');
break;
}
}
break;
}
});
callback();
},
function setupPeriodicPowerCheck(callback) {
if(config.subTopics.indexOf('power') !== -1) {
setInterval(function() {
async.forEachOf(config.idMap, function(_, id, callback) { reqPowerStatus(id, callback); });
}, config.powerCheckIntMs);
}
}
]);