|
|
|
@@ -14,6 +14,9 @@ New 2015-09-18:
|
|
|
|
New 2015-09-19:
|
|
|
|
New 2015-09-19:
|
|
|
|
- GarageDoorOpener Service
|
|
|
|
- GarageDoorOpener Service
|
|
|
|
- MotionSensor Service
|
|
|
|
- MotionSensor Service
|
|
|
|
|
|
|
|
New 2015-10-02:
|
|
|
|
|
|
|
|
- Check for valid group addresses
|
|
|
|
|
|
|
|
- new "R" flag allowed for Boolean addresses: 1/2/3R is the boolean not(1/2/3), i.e. 0 and 1 switched on read and write
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
var Service = require("HAP-NodeJS").Service;
|
|
|
|
var Service = require("HAP-NodeJS").Service;
|
|
|
|
@@ -24,6 +27,8 @@ var knxd_startMonitor = require('../platforms/KNX.js').startMonitor;
|
|
|
|
|
|
|
|
|
|
|
|
var milliTimeout = 300; // used to block responses while swiping
|
|
|
|
var milliTimeout = 300; // used to block responses while swiping
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var colorOn = "\x1b[30;47m";
|
|
|
|
|
|
|
|
var colorOff = "\x1b[0m";
|
|
|
|
|
|
|
|
|
|
|
|
function KNXDevice(log, config) {
|
|
|
|
function KNXDevice(log, config) {
|
|
|
|
this.log = log;
|
|
|
|
this.log = log;
|
|
|
|
@@ -115,6 +120,7 @@ KNXDevice.prototype = {
|
|
|
|
if (!groupAddress) {
|
|
|
|
if (!groupAddress) {
|
|
|
|
return null;
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.log("[knxdevice:knxread] preparing knx request for "+groupAddress);
|
|
|
|
var knxdConnection = new knxd.Connection();
|
|
|
|
var knxdConnection = new knxd.Connection();
|
|
|
|
// this.log("DEBUG in knxread: created empty connection, trying to connect socket to "+this.knxd_ip+":"+this.knxd_port);
|
|
|
|
// this.log("DEBUG in knxread: created empty connection, trying to connect socket to "+this.knxd_ip+":"+this.knxd_port);
|
|
|
|
knxdConnection.socketRemote({ host: this.knxd_ip, port: this.knxd_port }, function() {
|
|
|
|
knxdConnection.socketRemote({ host: this.knxd_ip, port: this.knxd_port }, function() {
|
|
|
|
@@ -130,7 +136,7 @@ KNXDevice.prototype = {
|
|
|
|
if (err) {
|
|
|
|
if (err) {
|
|
|
|
this.log("[ERROR] knxread:sendAPDU: " + err);
|
|
|
|
this.log("[ERROR] knxread:sendAPDU: " + err);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
this.log("knx request sent for "+groupAddress);
|
|
|
|
this.log("[knxdevice:knxread] knx request sent for "+groupAddress);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@@ -143,12 +149,12 @@ KNXDevice.prototype = {
|
|
|
|
// handle multiple addresses
|
|
|
|
// handle multiple addresses
|
|
|
|
for (var i = 0; i < groupAddresses.length; i++) {
|
|
|
|
for (var i = 0; i < groupAddresses.length; i++) {
|
|
|
|
if (groupAddresses[i]) { // do not bind empty addresses
|
|
|
|
if (groupAddresses[i]) { // do not bind empty addresses
|
|
|
|
this.knxread (groupAddresses[i]);
|
|
|
|
this.knxread (groupAddresses[i].match(/(\d*\/\d*\/\d*)/)[0]); // clean address
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
// it's only one
|
|
|
|
// it's only one
|
|
|
|
this.knxread (groupAddresses);
|
|
|
|
this.knxread (groupAddresses.match(/(\d*\/\d*\/\d*)/)[0]); // regex for cleaning address
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
@@ -158,70 +164,80 @@ KNXDevice.prototype = {
|
|
|
|
// boolean: get 0 or 1 from the bus, write boolean
|
|
|
|
// boolean: get 0 or 1 from the bus, write boolean
|
|
|
|
knxregister_bool: function(addresses, characteristic) {
|
|
|
|
knxregister_bool: function(addresses, characteristic) {
|
|
|
|
this.log("knx registering BOOLEAN " + addresses);
|
|
|
|
this.log("knx registering BOOLEAN " + addresses);
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type){
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type, reverse){
|
|
|
|
this.log("Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type + " for " + characteristic.displayName);
|
|
|
|
this.log("[" +this.name + "]: Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type + " for " + characteristic.displayName);
|
|
|
|
// iterate(characteristic);
|
|
|
|
// iterate(characteristic);
|
|
|
|
characteristic.setValue(val ? 1 : 0, undefined, 'fromKNXBus');
|
|
|
|
|
|
|
|
}.bind(this));
|
|
|
|
characteristic.setValue(val ? (reverse ? 0:1) : (reverse ? 1:0), undefined, 'fromKNXBus');
|
|
|
|
},
|
|
|
|
|
|
|
|
knxregister_boolReverse: function(addresses, characteristic) {
|
|
|
|
|
|
|
|
this.log("knx registering BOOLEAN " + addresses);
|
|
|
|
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type){
|
|
|
|
|
|
|
|
this.log("Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type + " for " + characteristic.displayName);
|
|
|
|
|
|
|
|
// iterate(characteristic);
|
|
|
|
|
|
|
|
characteristic.setValue(val ? 0 : 1, undefined, 'fromKNXBus');
|
|
|
|
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
// knxregister_boolReverse: function(addresses, characteristic) {
|
|
|
|
|
|
|
|
// this.log("knx registering BOOLEAN REVERSE " + addresses);
|
|
|
|
|
|
|
|
// knxd_registerGA(addresses, function(val, src, dest, type, reverse){
|
|
|
|
|
|
|
|
// this.log("[" +this.name + "]: Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type + " for " + characteristic.displayName);
|
|
|
|
|
|
|
|
//// iterate(characteristic);
|
|
|
|
|
|
|
|
// characteristic.setValue(val ? 0 : 1, undefined, 'fromKNXBus');
|
|
|
|
|
|
|
|
// }.bind(this));
|
|
|
|
|
|
|
|
// },
|
|
|
|
// percentage: get 0..255 from the bus, write 0..100 to characteristic
|
|
|
|
// percentage: get 0..255 from the bus, write 0..100 to characteristic
|
|
|
|
knxregister_percent: function(addresses, characteristic) {
|
|
|
|
knxregister_percent: function(addresses, characteristic) {
|
|
|
|
this.log("knx registering PERCENT " + addresses);
|
|
|
|
this.log("knx registering PERCENT " + addresses);
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type){
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type, reverse){
|
|
|
|
this.log("Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type+ " for " + characteristic.displayName);
|
|
|
|
this.log("[" +this.name + "]: Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type+ " for " + characteristic.displayName);
|
|
|
|
if (type !== "DPT5") {
|
|
|
|
if (type !== "DPT5") {
|
|
|
|
this.log("[ERROR] Received value cannot be a percentage value");
|
|
|
|
this.log("[ERROR] Received value cannot be a percentage value");
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
if (!characteristic.timeout) {
|
|
|
|
characteristic.setValue(Math.round(( reverse ? (255-val):val)/255*100), undefined, 'fromKNXBus');
|
|
|
|
if (characteristic.timeout < Date.now()) {
|
|
|
|
|
|
|
|
characteristic.setValue(Math.round(val/255*100), undefined, 'fromKNXBus');
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
this.log("Blackout time");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
characteristic.setValue(Math.round(val/255*100), undefined, 'fromKNXBus');
|
|
|
|
|
|
|
|
} // todo get the boolean logic right into one OR expresssion
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
},
|
|
|
|
},
|
|
|
|
// float
|
|
|
|
// float
|
|
|
|
knxregister_float: function(addresses, characteristic) {
|
|
|
|
knxregister_float: function(addresses, characteristic) {
|
|
|
|
this.log("knx registering FLOAT " + addresses);
|
|
|
|
// update for props refactor https://github.com/KhaosT/HAP-NodeJS/commit/1d84d128d1513beedcafc24d2c07d98185563243#diff-cb84de3a1478a38b2cf8388d709f1c1cR50
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type){
|
|
|
|
|
|
|
|
this.log("Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type+ " for " + characteristic.displayName);
|
|
|
|
var validValue = true;
|
|
|
|
var hk_value = Math.round(val*10)/10;
|
|
|
|
var hk_value = 0.0;
|
|
|
|
if (hk_value>=characteristic.minimumValue && hk_value<=characteristic.maximumValue) {
|
|
|
|
this.log("["+ this.name +"]:[" + characteristic.displayName+ "]:knx registering FLOAT " + addresses);
|
|
|
|
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type, reverse){
|
|
|
|
|
|
|
|
this.log("["+ this.name +"]:[" + characteristic.displayName+ "]: Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type+ " for " + characteristic.displayName);
|
|
|
|
|
|
|
|
// make hk_value compliant to properties
|
|
|
|
|
|
|
|
if (characteristic.props.minStep) {
|
|
|
|
|
|
|
|
// quantize
|
|
|
|
|
|
|
|
hk_value = Math.round(val/characteristic.props.minStep)/(1/characteristic.props.minStep);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
hk_value = val;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// range check
|
|
|
|
|
|
|
|
validValue = true; // assume validity at beginning
|
|
|
|
|
|
|
|
if (characteristic.props.minValue) {
|
|
|
|
|
|
|
|
validValue = validValue && (hk_value>=characteristic.props.minValue);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (characteristic.props.maxValue) {
|
|
|
|
|
|
|
|
validValue = validValue && (hk_value<=characteristic.props.maxValue);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (validValue) {
|
|
|
|
characteristic.setValue(hk_value, undefined, 'fromKNXBus'); // 1 decimal for HomeKit
|
|
|
|
characteristic.setValue(hk_value, undefined, 'fromKNXBus'); // 1 decimal for HomeKit
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
this.log("Value %s out of bounds %s...%s ",hk_value, characteristic.minimumValue, characteristic.maximumValue);
|
|
|
|
this.log("["+ this.name +"]:[" + characteristic.displayName+ "]: Value %s out of bounds %s...%s ",hk_value, characteristic.props.minValue, characteristic.props.maxValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
},
|
|
|
|
},
|
|
|
|
//integer
|
|
|
|
//integer
|
|
|
|
knxregister_int: function(addresses, characteristic) {
|
|
|
|
knxregister_int: function(addresses, characteristic) {
|
|
|
|
this.log("knx registering FLOAT " + addresses);
|
|
|
|
this.log("["+ this.name +"]:[" + characteristic.displayName+ "]:knx registering INT " + addresses);
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type){
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type, reverse){
|
|
|
|
this.log("Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type+ " for " + characteristic.displayName);
|
|
|
|
this.log("["+ this.name +"]:[" + characteristic.displayName+ "]: Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type+ " for " + characteristic.displayName);
|
|
|
|
if (val>=(characteristic.minimumValue || 0) && val<=(characteristic.maximumValue || 255)) {
|
|
|
|
if (val>=(characteristic.props.minValue || 0) && val<=(characteristic.props.maxValue || 255)) {
|
|
|
|
characteristic.setValue(val, undefined, 'fromKNXBus');
|
|
|
|
characteristic.setValue(reverse ? (255-val):val, undefined, 'fromKNXBus');
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
this.log("Value %s out of bounds %s...%s ",hk_value, (characteristic.minimumValue || 0), (characteristic.maximumValue || 255));
|
|
|
|
this.log("["+ this.name +"]:[" + characteristic.displayName+ "]: Value %s out of bounds %s...%s ",hk_value, (characteristic.props.minValue || 0), (characteristic.props.maxValue || 255));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
},
|
|
|
|
},
|
|
|
|
knxregister_HVAC: function(addresses, characteristic) {
|
|
|
|
knxregister_HVAC: function(addresses, characteristic) {
|
|
|
|
this.log("knx registering HVAC " + addresses);
|
|
|
|
this.log("["+ this.name +"]:[" + characteristic.displayName+ "]:knx registering HVAC " + addresses);
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type){
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type){
|
|
|
|
this.log("Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type+ " for " + characteristic.displayName);
|
|
|
|
this.log("["+ this.name +"]:[" + characteristic.displayName+ "]:Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type+ " for " + characteristic.displayName);
|
|
|
|
var HAPvalue = 0;
|
|
|
|
var HAPvalue = 0;
|
|
|
|
switch (val){
|
|
|
|
switch (val){
|
|
|
|
case 0:
|
|
|
|
case 0:
|
|
|
|
@@ -261,9 +277,9 @@ KNXDevice.prototype = {
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
// undefined, has to match!
|
|
|
|
// undefined, has to match!
|
|
|
|
knxregister: function(addresses, characteristic) {
|
|
|
|
knxregister: function(addresses, characteristic) {
|
|
|
|
this.log("knx registering " + addresses);
|
|
|
|
this.log("["+ this.name +"]:[" + characteristic.displayName+ "]:knx registering " + addresses);
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type){
|
|
|
|
knxd_registerGA(addresses, function(val, src, dest, type){
|
|
|
|
this.log("Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type+ " for " + characteristic.displayName);
|
|
|
|
this.log("["+ this.name +"]:[" + characteristic.displayName+ "]:Received value from bus:"+val+ " for " +dest+ " from "+src+" of type "+type+ " for " + characteristic.displayName);
|
|
|
|
characteristic.setValue(val, undefined, 'fromKNXBus');
|
|
|
|
characteristic.setValue(val, undefined, 'fromKNXBus');
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
},
|
|
|
|
},
|
|
|
|
@@ -276,71 +292,74 @@ KNXDevice.prototype = {
|
|
|
|
* }.bind(this));
|
|
|
|
* }.bind(this));
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
setBooleanState: function(value, callback, context, gaddress) {
|
|
|
|
setBooleanState: function(value, callback, context, gaddress, reverseflag) {
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
this.log(gaddress + " event ping pong, exit!");
|
|
|
|
// this.log(gaddress + " event ping pong, exit!");
|
|
|
|
if (callback) {
|
|
|
|
if (callback) {
|
|
|
|
callback();
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
var numericValue = 0;
|
|
|
|
var numericValue = reverseflag ? 1:0;
|
|
|
|
if (value) {
|
|
|
|
if (value) {
|
|
|
|
numericValue = 1; // need 0 or 1, not true or something
|
|
|
|
numericValue = reverseflag ? 0:1; // need 0 or 1, not true or something
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.log("Setting "+gaddress+" Boolean to %s", numericValue);
|
|
|
|
this.log("["+ this.name +"]:Setting "+gaddress+" " + reverseflag ? " (reverse)":""+ " Boolean to %s", numericValue);
|
|
|
|
this.knxwrite(callback, gaddress,'DPT1',numericValue);
|
|
|
|
this.knxwrite(callback, gaddress,'DPT1',numericValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
},
|
|
|
|
},
|
|
|
|
setBooleanReverseState: function(value, callback, context, gaddress) {
|
|
|
|
// setBooleanReverseState: function(value, callback, context, gaddress) {
|
|
|
|
|
|
|
|
// if (context === 'fromKNXBus') {
|
|
|
|
|
|
|
|
//// this.log(gaddress + " event ping pong, exit!");
|
|
|
|
|
|
|
|
// if (callback) {
|
|
|
|
|
|
|
|
// callback();
|
|
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
// } else {
|
|
|
|
|
|
|
|
// var numericValue = 0;
|
|
|
|
|
|
|
|
// if (!value) {
|
|
|
|
|
|
|
|
// numericValue = 1; // need 0 or 1, not true or something
|
|
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
// this.log("["+ this.name +"]:Setting "+gaddress+" Boolean to %s", numericValue);
|
|
|
|
|
|
|
|
// this.knxwrite(callback, gaddress,'DPT1',numericValue);
|
|
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
// },
|
|
|
|
|
|
|
|
setPercentage: function(value, callback, context, gaddress, reverseflag) {
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
this.log(gaddress + " event ping pong, exit!");
|
|
|
|
// this.log(gaddress + "event ping pong, exit!");
|
|
|
|
if (callback) {
|
|
|
|
if (callback) {
|
|
|
|
callback();
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
var numericValue = 0;
|
|
|
|
var numericValue = 0;
|
|
|
|
if (!value) {
|
|
|
|
value = ( value>=0 ? (value<=100 ? value:100):0 ); //ensure range 0..100
|
|
|
|
numericValue = 1; // need 0 or 1, not true or something
|
|
|
|
if (reverseflag) {
|
|
|
|
}
|
|
|
|
numericValue = 255 - Math.round(255*value/100); // convert 0..100 to 255..0 for KNX bus
|
|
|
|
this.log("Setting "+gaddress+" Boolean to %s", numericValue);
|
|
|
|
|
|
|
|
this.knxwrite(callback, gaddress,'DPT1',numericValue);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
setPercentage: function(value, callback, context, gaddress) {
|
|
|
|
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
|
|
|
|
this.log("event ping pong, exit!");
|
|
|
|
|
|
|
|
if (callback) {
|
|
|
|
|
|
|
|
callback();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
var numericValue = 0;
|
|
|
|
numericValue = Math.round(255*value/100); // convert 0..100 to 0..255 for KNX bus
|
|
|
|
if (value) {
|
|
|
|
|
|
|
|
numericValue = Math.round(255*value/100); // convert 1..100 to 1..255 for KNX bus
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.log("Setting "+gaddress+" percentage to %s (%s)", value, numericValue);
|
|
|
|
this.log("["+ this.name +"]:Setting "+gaddress+" percentage to %s (%s)", value, numericValue);
|
|
|
|
this.knxwrite(callback, gaddress,'DPT5',numericValue);
|
|
|
|
this.knxwrite(callback, gaddress,'DPT5',numericValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
setInt: function(value, callback, context, gaddress) {
|
|
|
|
setInt: function(value, callback, context, gaddress) {
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
this.log("event ping pong, exit!");
|
|
|
|
// this.log(gaddress + "event ping pong, exit!");
|
|
|
|
if (callback) {
|
|
|
|
if (callback) {
|
|
|
|
callback();
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
var numericValue = 0;
|
|
|
|
var numericValue = 0;
|
|
|
|
if (value && value>=0 && value<=255) {
|
|
|
|
if (value && value>=0 && value<=255) {
|
|
|
|
numericValue = value; // assure 1..255 for KNX bus
|
|
|
|
numericValue = value; // assure 0..255 for KNX bus
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.log("Setting "+gaddress+" int to %s (%s)", value, numericValue);
|
|
|
|
this.log("["+ this.name +"]:Setting "+gaddress+" int to %s (%s)", value, numericValue);
|
|
|
|
this.knxwrite(callback, gaddress,'DPT5',numericValue);
|
|
|
|
this.knxwrite(callback, gaddress,'DPT5',numericValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
setFloat: function(value, callback, context, gaddress) {
|
|
|
|
setFloat: function(value, callback, context, gaddress) {
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
this.log(gaddress + " event ping pong, exit!");
|
|
|
|
// this.log(gaddress + " event ping pong, exit!");
|
|
|
|
if (callback) {
|
|
|
|
if (callback) {
|
|
|
|
callback();
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@@ -349,13 +368,13 @@ KNXDevice.prototype = {
|
|
|
|
if (value) {
|
|
|
|
if (value) {
|
|
|
|
numericValue = value; // homekit expects precision of 1 decimal
|
|
|
|
numericValue = value; // homekit expects precision of 1 decimal
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.log("Setting "+gaddress+" Float to %s", numericValue);
|
|
|
|
this.log("["+ this.name +"]:Setting "+gaddress+" Float to %s", numericValue);
|
|
|
|
this.knxwrite(callback, gaddress,'DPT9',numericValue);
|
|
|
|
this.knxwrite(callback, gaddress,'DPT9',numericValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
setHVACState: function(value, callback, context, gaddress) {
|
|
|
|
setHVACState: function(value, callback, context, gaddress) {
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
if (context === 'fromKNXBus') {
|
|
|
|
this.log(gaddress + " event ping pong, exit!");
|
|
|
|
// this.log(gaddress + " event ping pong, exit!");
|
|
|
|
if (callback) {
|
|
|
|
if (callback) {
|
|
|
|
callback();
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@@ -378,7 +397,7 @@ KNXDevice.prototype = {
|
|
|
|
KNXvalue = 1;
|
|
|
|
KNXvalue = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
this.log("Setting "+gaddress+" HVAC to %s", KNXvalue);
|
|
|
|
this.log("["+ this.name +"]:Setting "+gaddress+" HVAC to %s", KNXvalue);
|
|
|
|
this.knxwrite(callback, gaddress,'DPT5',KNXvalue);
|
|
|
|
this.knxwrite(callback, gaddress,'DPT5',KNXvalue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@@ -387,33 +406,50 @@ KNXDevice.prototype = {
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
identify: function(callback) {
|
|
|
|
identify: function(callback) {
|
|
|
|
this.log("Identify requested!");
|
|
|
|
this.log("["+ this.name +"]:Identify requested!");
|
|
|
|
callback(); // success
|
|
|
|
callback(); // success
|
|
|
|
},
|
|
|
|
},
|
|
|
|
/** bindCharacteristic
|
|
|
|
/** bindCharacteristic
|
|
|
|
* initializes callbacks for 'set' events (from HK) and for KNX bus reads (to HK)
|
|
|
|
* initializes callbacks for 'set' events (from HK) and for KNX bus reads (to HK)
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
bindCharacteristic: function(myService, characteristicType, valueType, config) {
|
|
|
|
bindCharacteristic: function(myService, characteristicType, valueType, config, defaultValue) {
|
|
|
|
var myCharacteristic = myService.getCharacteristic(characteristicType);
|
|
|
|
var myCharacteristic = myService.getCharacteristic(characteristicType);
|
|
|
|
|
|
|
|
var setGA = "";
|
|
|
|
|
|
|
|
var setReverse = false;
|
|
|
|
if (myCharacteristic === undefined) {
|
|
|
|
if (myCharacteristic === undefined) {
|
|
|
|
throw new Error("unknown characteristics cannot be bound");
|
|
|
|
throw new Error("unknown characteristics cannot be bound");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (defaultValue) {
|
|
|
|
|
|
|
|
myCharacteristic.setValue(defaultValue);
|
|
|
|
|
|
|
|
}
|
|
|
|
if (config.Set) {
|
|
|
|
if (config.Set) {
|
|
|
|
// can write
|
|
|
|
// can write
|
|
|
|
|
|
|
|
// extract address and Reverse flag
|
|
|
|
|
|
|
|
setGA = config.Set.match(/\d*\/\d*\/\d*/);
|
|
|
|
|
|
|
|
if (setGA===null) {
|
|
|
|
|
|
|
|
this.log(colorOn + "["+ this.name +"]:["+myCharacteristic.displayName+"] Error in group adress: ["+ config.Set +"] "+colorOff);
|
|
|
|
|
|
|
|
throw new Error("EINVGROUPADRESS - Invalid group address given");
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
setGA=setGA[0]; // first element of returned array is the group address
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
setReverse = config.Set.match(/\d*\/\d*\/\d*(R)/) ? true:false;
|
|
|
|
|
|
|
|
|
|
|
|
switch (valueType) {
|
|
|
|
switch (valueType) {
|
|
|
|
case "Bool":
|
|
|
|
case "Bool":
|
|
|
|
myCharacteristic.on('set', function(value, callback, context) {
|
|
|
|
myCharacteristic.on('set', function(value, callback, context) {
|
|
|
|
this.setBooleanState(value, callback, context, config.Set);
|
|
|
|
this.setBooleanState(value, callback, context, setGA, setReverse); //NEW
|
|
|
|
}.bind(this));
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "BoolReverse":
|
|
|
|
|
|
|
|
myCharacteristic.on('set', function(value, callback, context) {
|
|
|
|
|
|
|
|
this.setBooleanReverseState(value, callback, context, config.Set);
|
|
|
|
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
// case "BoolReverse":
|
|
|
|
|
|
|
|
// this.log("["+ this.name +"]:["+myCharacteristic.displayName+"] \x1b[30;47m%s\x1b[0mWARNING in group adress: "+ config.Set +": Legacy BoolReverse used. Use " + config.Set +"R instead");
|
|
|
|
|
|
|
|
// myCharacteristic.on('set', function(value, callback, context) {
|
|
|
|
|
|
|
|
// this.setBooleanReverseState(value, callback, context, config.Set);
|
|
|
|
|
|
|
|
// }.bind(this));
|
|
|
|
|
|
|
|
// break;
|
|
|
|
case "Percent":
|
|
|
|
case "Percent":
|
|
|
|
myCharacteristic.on('set', function(value, callback, context) {
|
|
|
|
myCharacteristic.on('set', function(value, callback, context) {
|
|
|
|
this.setPercentage(value, callback, context, config.Set);
|
|
|
|
this.setPercentage(value, callback, context, setGA, setReverse);
|
|
|
|
myCharacteristic.timeout = Date.now()+milliTimeout;
|
|
|
|
myCharacteristic.timeout = Date.now()+milliTimeout;
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
@@ -433,7 +469,7 @@ KNXDevice.prototype = {
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
default: {
|
|
|
|
default: {
|
|
|
|
this.log("[ERROR] unknown type passed");
|
|
|
|
this.log(colorOn + "[ERROR] unknown type passed: [" + valueType+"]"+ colorOff);
|
|
|
|
throw new Error("[ERROR] unknown type passed");
|
|
|
|
throw new Error("[ERROR] unknown type passed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@@ -445,9 +481,9 @@ KNXDevice.prototype = {
|
|
|
|
case "Bool":
|
|
|
|
case "Bool":
|
|
|
|
this.knxregister_bool([config.Set].concat(config.Listen || []), myCharacteristic);
|
|
|
|
this.knxregister_bool([config.Set].concat(config.Listen || []), myCharacteristic);
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
case "BoolReverse":
|
|
|
|
// case "BoolReverse":
|
|
|
|
this.knxregister_boolReverse([config.Set].concat(config.Listen || []), myCharacteristic);
|
|
|
|
// this.knxregister_boolReverse([config.Set].concat(config.Listen || []), myCharacteristic);
|
|
|
|
break;
|
|
|
|
// break;
|
|
|
|
case "Percent":
|
|
|
|
case "Percent":
|
|
|
|
this.knxregister_percent([config.Set].concat(config.Listen || []), myCharacteristic);
|
|
|
|
this.knxregister_percent([config.Set].concat(config.Listen || []), myCharacteristic);
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
@@ -461,10 +497,10 @@ KNXDevice.prototype = {
|
|
|
|
this.knxregister_HVAC([config.Set].concat(config.Listen || []), myCharacteristic);
|
|
|
|
this.knxregister_HVAC([config.Set].concat(config.Listen || []), myCharacteristic);
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
default:
|
|
|
|
this.log("[ERROR] unknown type passed");
|
|
|
|
this.log(colorOn+ "[ERROR] unknown type passed: ["+valueType+"]"+colorOff);
|
|
|
|
throw new Error("[ERROR] unknown type passed");
|
|
|
|
throw new Error("[ERROR] unknown type passed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.log("Issuing read requests on the KNX bus...");
|
|
|
|
this.log("["+ this.name +"]:["+myCharacteristic.displayName+"]: Issuing read requests on the KNX bus...");
|
|
|
|
this.knxreadarray([config.Set].concat(config.Listen || []));
|
|
|
|
this.knxreadarray([config.Set].concat(config.Listen || []));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return myCharacteristic; // for chaining or whatsoever
|
|
|
|
return myCharacteristic; // for chaining or whatsoever
|
|
|
|
@@ -494,30 +530,30 @@ KNXDevice.prototype = {
|
|
|
|
|
|
|
|
|
|
|
|
var myService = new Service.ContactSensor(config.name,config.name);
|
|
|
|
var myService = new Service.ContactSensor(config.name,config.name);
|
|
|
|
if (config.ContactSensorState) {
|
|
|
|
if (config.ContactSensorState) {
|
|
|
|
this.log("ContactSensor ContactSensorState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:ContactSensor ContactSensorState characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.ContactSensorState, "Bool", config.ContactSensorState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.ContactSensorState, "Bool", config.ContactSensorState);
|
|
|
|
} else if (config.ContactSensorStateContact1) {
|
|
|
|
} else if (config.ContactSensorStateContact1) {
|
|
|
|
this.log("ContactSensor ContactSensorStateContact1 characteristic enabled");
|
|
|
|
this.log(colorOn+ "[ERROR] outdated type passed: [ContactSensorStateContact1]"+colorOff);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.ContactSensorState, "BoolReverse", config.ContactSensorStateContact1);
|
|
|
|
throw new Error("[ERROR] outdated type passed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//optionals
|
|
|
|
//optionals
|
|
|
|
if (config.StatusActive) {
|
|
|
|
if (config.StatusActive) {
|
|
|
|
this.log("ContactSensor StatusActive characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:ContactSensor StatusActive characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.StatusActive);
|
|
|
|
myService.addCharacteristic(Characteristic.StatusActive);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusActive, "Bool", config.StatusActive);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusActive, "Bool", config.StatusActive);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.StatusFault) {
|
|
|
|
if (config.StatusFault) {
|
|
|
|
this.log("ContactSensor StatusFault characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:ContactSensor StatusFault characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.StatusFault);
|
|
|
|
myService.addCharacteristic(Characteristic.StatusFault);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusFault, "Bool", config.StatusFault);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusFault, "Bool", config.StatusFault);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.StatusTampered) {
|
|
|
|
if (config.StatusTampered) {
|
|
|
|
this.log("ContactSensor StatusTampered characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:ContactSensor StatusTampered characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.StatusTampered);
|
|
|
|
myService.addCharacteristic(Characteristic.StatusTampered);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusTampered, "Bool", config.StatusTampered);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusTampered, "Bool", config.StatusTampered);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.StatusLowBattery) {
|
|
|
|
if (config.StatusLowBattery) {
|
|
|
|
this.log("ContactSensor StatusLowBattery characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:ContactSensor StatusLowBattery characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.StatusLowBattery);
|
|
|
|
myService.addCharacteristic(Characteristic.StatusLowBattery);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusLowBattery, "Bool", config.StatusLowBattery);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusLowBattery, "Bool", config.StatusLowBattery);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@@ -555,27 +591,27 @@ KNXDevice.prototype = {
|
|
|
|
|
|
|
|
|
|
|
|
var myService = new Service.GarageDoorOpener(config.name,config.name);
|
|
|
|
var myService = new Service.GarageDoorOpener(config.name,config.name);
|
|
|
|
if (config.CurrentDoorState) {
|
|
|
|
if (config.CurrentDoorState) {
|
|
|
|
this.log("GarageDoorOpener CurrentDoorState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:GarageDoorOpener CurrentDoorState characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentDoorState, "Int", config.CurrentDoorState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentDoorState, "Int", config.CurrentDoorState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.TargetDoorState) {
|
|
|
|
if (config.TargetDoorState) {
|
|
|
|
this.log("GarageDoorOpener TargetDoorState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:GarageDoorOpener TargetDoorState characteristic enabled");
|
|
|
|
//myService.getCharacteristic(Characteristic.TargetDoorState).minimumValue=0; //
|
|
|
|
//myService.getCharacteristic(Characteristic.TargetDoorState).minimumValue=0; //
|
|
|
|
//myService.getCharacteristic(Characteristic.TargetDoorState).maximumValue=4; //
|
|
|
|
//myService.getCharacteristic(Characteristic.TargetDoorState).maximumValue=4; //
|
|
|
|
this.bindCharacteristic(myService, Characteristic.TargetDoorState, "Int", config.TargetDoorState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.TargetDoorState, "Int", config.TargetDoorState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.ObstructionDetected) {
|
|
|
|
if (config.ObstructionDetected) {
|
|
|
|
this.log("GarageDoorOpener ObstructionDetected characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:GarageDoorOpener ObstructionDetected characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.ObstructionDetected, "Bool", config.ObstructionDetected);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.ObstructionDetected, "Bool", config.ObstructionDetected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//optionals
|
|
|
|
//optionals
|
|
|
|
if (config.LockCurrentState) {
|
|
|
|
if (config.LockCurrentState) {
|
|
|
|
this.log("GarageDoorOpener LockCurrentState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:GarageDoorOpener LockCurrentState characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.LockCurrentState);
|
|
|
|
myService.addCharacteristic(Characteristic.LockCurrentState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.LockCurrentState, "Int", config.LockCurrentState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.LockCurrentState, "Int", config.LockCurrentState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.LockTargetState) {
|
|
|
|
if (config.LockTargetState) {
|
|
|
|
this.log("GarageDoorOpener LockTargetState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:GarageDoorOpener LockTargetState characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.LockTargetState);
|
|
|
|
myService.addCharacteristic(Characteristic.LockTargetState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.LockTargetState, "Bool", config.LockTargetState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.LockTargetState, "Bool", config.LockTargetState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@@ -596,12 +632,12 @@ KNXDevice.prototype = {
|
|
|
|
var myService = new Service.Lightbulb(config.name,config.name);
|
|
|
|
var myService = new Service.Lightbulb(config.name,config.name);
|
|
|
|
// On (and Off)
|
|
|
|
// On (and Off)
|
|
|
|
if (config.On) {
|
|
|
|
if (config.On) {
|
|
|
|
this.log("Lightbulb on/off characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Lightbulb on/off characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.On, "Bool", config.On);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.On, "Bool", config.On);
|
|
|
|
} // On characteristic
|
|
|
|
} // On characteristic
|
|
|
|
// Brightness if available
|
|
|
|
// Brightness if available
|
|
|
|
if (config.Brightness) {
|
|
|
|
if (config.Brightness) {
|
|
|
|
this.log("Lightbulb Brightness characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Lightbulb Brightness characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.Brightness); // it's an optional
|
|
|
|
myService.addCharacteristic(Characteristic.Brightness); // it's an optional
|
|
|
|
this.bindCharacteristic(myService, Characteristic.Brightness, "Percent", config.Brightness);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.Brightness, "Percent", config.Brightness);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@@ -623,7 +659,7 @@ KNXDevice.prototype = {
|
|
|
|
var myService = new Service.LightSensor(config.name,config.name);
|
|
|
|
var myService = new Service.LightSensor(config.name,config.name);
|
|
|
|
// CurrentTemperature)
|
|
|
|
// CurrentTemperature)
|
|
|
|
if (config.CurrentAmbientLightLevel) {
|
|
|
|
if (config.CurrentAmbientLightLevel) {
|
|
|
|
this.log("LightSensor CurrentAmbientLightLevel characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:LightSensor CurrentAmbientLightLevel characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentAmbientLightLevel, "Float", config.CurrentAmbientLightLevel);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentAmbientLightLevel, "Float", config.CurrentAmbientLightLevel);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return myService;
|
|
|
|
return myService;
|
|
|
|
@@ -648,20 +684,20 @@ KNXDevice.prototype = {
|
|
|
|
// LockCurrentState
|
|
|
|
// LockCurrentState
|
|
|
|
if (config.LockCurrentState) {
|
|
|
|
if (config.LockCurrentState) {
|
|
|
|
// for normal contacts: Secured = 1
|
|
|
|
// for normal contacts: Secured = 1
|
|
|
|
this.log("LockMechanism LockCurrentState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:LockMechanism LockCurrentState characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.LockCurrentState, "Bool", config.LockCurrentState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.LockCurrentState, "Bool", config.LockCurrentState);
|
|
|
|
} else if (config.LockCurrentStateSecured0) {
|
|
|
|
} else if (config.LockCurrentStateSecured0) {
|
|
|
|
// for reverse contacts Secured = 0
|
|
|
|
// for reverse contacts Secured = 0
|
|
|
|
this.log("LockMechanism LockCurrentState characteristic enabled");
|
|
|
|
this.log(colorOn+ "[ERROR] outdated type passed: [LockCurrentStateSecured0]"+colorOff);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.LockCurrentState, "BoolReverse", config.LockCurrentStateSecured0);
|
|
|
|
throw new Error("[ERROR] outdated type passed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// LockTargetState
|
|
|
|
// LockTargetState
|
|
|
|
if (config.LockTargetState) {
|
|
|
|
if (config.LockTargetState) {
|
|
|
|
this.log("LockMechanism LockTargetState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:LockMechanism LockTargetState characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.LockTargetState, "Bool", config.LockTargetState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.LockTargetState, "Bool", config.LockTargetState);
|
|
|
|
} else if (config.LockTargetStateSecured0) {
|
|
|
|
} else if (config.LockTargetStateSecured0) {
|
|
|
|
this.log("LockMechanism LockTargetState characteristic enabled");
|
|
|
|
this.log(colorOn+ "[ERROR] outdated type passed: [LockTargetStateSecured0]"+colorOff);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.LockTargetState, "BoolReverse", config.LockTargetStateSecured0);
|
|
|
|
throw new Error("[ERROR] outdated type passed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//iterate(myService);
|
|
|
|
//iterate(myService);
|
|
|
|
@@ -683,27 +719,27 @@ KNXDevice.prototype = {
|
|
|
|
|
|
|
|
|
|
|
|
var myService = new Service.MotionSensor(config.name,config.name);
|
|
|
|
var myService = new Service.MotionSensor(config.name,config.name);
|
|
|
|
if (config.MotionDetected) {
|
|
|
|
if (config.MotionDetected) {
|
|
|
|
this.log("MotionSensor MotionDetected characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:MotionSensor MotionDetected characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.MotionDetected, "Bool", config.MotionDetected);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.MotionDetected, "Bool", config.MotionDetected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//optionals
|
|
|
|
//optionals
|
|
|
|
if (config.StatusActive) {
|
|
|
|
if (config.StatusActive) {
|
|
|
|
this.log("MotionSensor StatusActive characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:MotionSensor StatusActive characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.StatusActive);
|
|
|
|
myService.addCharacteristic(Characteristic.StatusActive);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusActive, "Bool", config.StatusActive);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusActive, "Bool", config.StatusActive);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.StatusFault) {
|
|
|
|
if (config.StatusFault) {
|
|
|
|
this.log("MotionSensor StatusFault characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:MotionSensor StatusFault characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.StatusFault);
|
|
|
|
myService.addCharacteristic(Characteristic.StatusFault);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusFault, "Bool", config.StatusFault);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusFault, "Bool", config.StatusFault);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.StatusTampered) {
|
|
|
|
if (config.StatusTampered) {
|
|
|
|
this.log("MotionSensor StatusTampered characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:MotionSensor StatusTampered characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.StatusTampered);
|
|
|
|
myService.addCharacteristic(Characteristic.StatusTampered);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusTampered, "Bool", config.StatusTampered);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusTampered, "Bool", config.StatusTampered);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.StatusLowBattery) {
|
|
|
|
if (config.StatusLowBattery) {
|
|
|
|
this.log("MotionSensor StatusLowBattery characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:MotionSensor StatusLowBattery characteristic enabled");
|
|
|
|
myService.addCharacteristic(Characteristic.StatusLowBattery);
|
|
|
|
myService.addCharacteristic(Characteristic.StatusLowBattery);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusLowBattery, "Bool", config.StatusLowBattery);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.StatusLowBattery, "Bool", config.StatusLowBattery);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@@ -726,11 +762,11 @@ KNXDevice.prototype = {
|
|
|
|
var myService = new Service.Outlet(config.name,config.name);
|
|
|
|
var myService = new Service.Outlet(config.name,config.name);
|
|
|
|
// On (and Off)
|
|
|
|
// On (and Off)
|
|
|
|
if (config.On) {
|
|
|
|
if (config.On) {
|
|
|
|
this.log("Outlet on/off characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Outlet on/off characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.On, "Bool", config.On);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.On, "Bool", config.On);
|
|
|
|
} // OutletInUse characteristic
|
|
|
|
} // OutletInUse characteristic
|
|
|
|
if (config.OutletInUse) {
|
|
|
|
if (config.OutletInUse) {
|
|
|
|
this.log("Outlet on/off characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Outlet on/off characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.OutletInUse, "Bool", config.OutletInUse);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.OutletInUse, "Bool", config.OutletInUse);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return myService;
|
|
|
|
return myService;
|
|
|
|
@@ -748,7 +784,7 @@ KNXDevice.prototype = {
|
|
|
|
var myService = new Service.Switch(config.name,config.name);
|
|
|
|
var myService = new Service.Switch(config.name,config.name);
|
|
|
|
// On (and Off)
|
|
|
|
// On (and Off)
|
|
|
|
if (config.On) {
|
|
|
|
if (config.On) {
|
|
|
|
this.log("Switch on/off characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Switch on/off characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.On, "Bool", config.On);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.On, "Bool", config.On);
|
|
|
|
} // On characteristic
|
|
|
|
} // On characteristic
|
|
|
|
|
|
|
|
|
|
|
|
@@ -775,26 +811,35 @@ KNXDevice.prototype = {
|
|
|
|
|
|
|
|
|
|
|
|
var myService = new Service.Thermostat(config.name,config.name);
|
|
|
|
var myService = new Service.Thermostat(config.name,config.name);
|
|
|
|
// CurrentTemperature)
|
|
|
|
// CurrentTemperature)
|
|
|
|
|
|
|
|
// props update for https://github.com/KhaosT/HAP-NodeJS/commit/1d84d128d1513beedcafc24d2c07d98185563243#diff-cb84de3a1478a38b2cf8388d709f1c1cR108
|
|
|
|
if (config.CurrentTemperature) {
|
|
|
|
if (config.CurrentTemperature) {
|
|
|
|
this.log("Thermostat CurrentTemperature characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Thermostat CurrentTemperature characteristic enabled");
|
|
|
|
|
|
|
|
myService.getCharacteristic(Characteristic.CurrentTemperature).setProps({
|
|
|
|
|
|
|
|
minValue: config.CurrentTemperature.minValue || -40,
|
|
|
|
|
|
|
|
maxValue: config.CurrentTemperature.maxValue || 60
|
|
|
|
|
|
|
|
}); // °C by default
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentTemperature, "Float", config.CurrentTemperature);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentTemperature, "Float", config.CurrentTemperature);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// TargetTemperature if available
|
|
|
|
// TargetTemperature if available
|
|
|
|
if (config.TargetTemperature) {
|
|
|
|
if (config.TargetTemperature) {
|
|
|
|
this.log("Thermostat TargetTemperature characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Thermostat TargetTemperature characteristic enabled");
|
|
|
|
// default boundary too narrow for thermostats
|
|
|
|
// default boundary too narrow for thermostats
|
|
|
|
myService.getCharacteristic(Characteristic.TargetTemperature).minimumValue=0; // °C
|
|
|
|
// props update for https://github.com/KhaosT/HAP-NodeJS/commit/1d84d128d1513beedcafc24d2c07d98185563243#diff-cb84de3a1478a38b2cf8388d709f1c1cR108
|
|
|
|
myService.getCharacteristic(Characteristic.TargetTemperature).maximumValue=40; // °C
|
|
|
|
myService.getCharacteristic(Characteristic.TargetTemperature).setProps({
|
|
|
|
|
|
|
|
minValue: config.TargetTemperature.minValue || 0,
|
|
|
|
|
|
|
|
maxValue: config.TargetTemperature.maxValue || 40
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.bindCharacteristic(myService, Characteristic.TargetTemperature, "Float", config.TargetTemperature);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.TargetTemperature, "Float", config.TargetTemperature);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// HVAC
|
|
|
|
// HVAC
|
|
|
|
if (config.CurrentHeatingCoolingState) {
|
|
|
|
if (config.CurrentHeatingCoolingState) {
|
|
|
|
this.log("Thermostat CurrentHeatingCoolingState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Thermostat CurrentHeatingCoolingState characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentHeatingCoolingState, "HVAC", config.CurrentHeatingCoolingState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentHeatingCoolingState, "HVAC", config.CurrentHeatingCoolingState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// HVAC
|
|
|
|
// HVAC
|
|
|
|
if (config.TargetHeatingCoolingState) {
|
|
|
|
if (config.TargetHeatingCoolingState) {
|
|
|
|
this.log("Thermostat TargetHeatingCoolingState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Thermostat TargetHeatingCoolingState characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.TargetHeatingCoolingState, "HVAC", config.TargetHeatingCoolingState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.TargetHeatingCoolingState, "HVAC", config.TargetHeatingCoolingState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return myService;
|
|
|
|
return myService;
|
|
|
|
@@ -812,10 +857,16 @@ KNXDevice.prototype = {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var myService = new Service.TemperatureSensor(config.name,config.name);
|
|
|
|
var myService = new Service.TemperatureSensor(config.name,config.name);
|
|
|
|
// CurrentTemperature)
|
|
|
|
// CurrentTemperature)
|
|
|
|
|
|
|
|
// props update for https://github.com/KhaosT/HAP-NodeJS/commit/1d84d128d1513beedcafc24d2c07d98185563243#diff-cb84de3a1478a38b2cf8388d709f1c1cR108
|
|
|
|
if (config.CurrentTemperature) {
|
|
|
|
if (config.CurrentTemperature) {
|
|
|
|
this.log("TemperatureSensor CurrentTemperature characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:TemperatureSensor CurrentTemperature characteristic enabled");
|
|
|
|
|
|
|
|
myService.getCharacteristic(Characteristic.CurrentTemperature).setProps({
|
|
|
|
|
|
|
|
minValue: config.CurrentTemperature.minValue || -40,
|
|
|
|
|
|
|
|
maxValue: config.CurrentTemperature.maxValue || 60
|
|
|
|
|
|
|
|
}); // °C by default
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentTemperature, "Float", config.CurrentTemperature);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentTemperature, "Float", config.CurrentTemperature);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return myService;
|
|
|
|
return myService;
|
|
|
|
},
|
|
|
|
},
|
|
|
|
getWindowService: function(config) {
|
|
|
|
getWindowService: function(config) {
|
|
|
|
@@ -845,16 +896,16 @@ KNXDevice.prototype = {
|
|
|
|
var myService = new Service.Window(config.name,config.name);
|
|
|
|
var myService = new Service.Window(config.name,config.name);
|
|
|
|
|
|
|
|
|
|
|
|
if (config.CurrentPosition) {
|
|
|
|
if (config.CurrentPosition) {
|
|
|
|
this.log("Window CurrentPosition characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Window CurrentPosition characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentPosition, "Percent", config.CurrentPosition);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentPosition, "Percent", config.CurrentPosition);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.TargetPosition) {
|
|
|
|
if (config.TargetPosition) {
|
|
|
|
this.log("Window TargetPosition characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Window TargetPosition characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.TargetPosition, "Percent", config.TargetPosition);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.TargetPosition, "Percent", config.TargetPosition);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.PositionState) {
|
|
|
|
if (config.PositionState) {
|
|
|
|
this.log("Window PositionState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:Window PositionState characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.PositionState, "Float", config.PositionState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.PositionState, "Int", config.PositionState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return myService;
|
|
|
|
return myService;
|
|
|
|
},
|
|
|
|
},
|
|
|
|
@@ -880,15 +931,15 @@ KNXDevice.prototype = {
|
|
|
|
|
|
|
|
|
|
|
|
var myService = new Service.WindowCovering(config.name,config.name);
|
|
|
|
var myService = new Service.WindowCovering(config.name,config.name);
|
|
|
|
if (config.CurrentPosition) {
|
|
|
|
if (config.CurrentPosition) {
|
|
|
|
this.log("WindowCovering CurrentPosition characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:WindowCovering CurrentPosition characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentPosition, "Percent", config.CurrentPosition);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.CurrentPosition, "Percent", config.CurrentPosition);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.TargetPosition) {
|
|
|
|
if (config.TargetPosition) {
|
|
|
|
this.log("WindowCovering TargetPosition characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:WindowCovering TargetPosition characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.TargetPosition, "Percent", config.TargetPosition);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.TargetPosition, "Percent", config.TargetPosition);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (config.PositionState) {
|
|
|
|
if (config.PositionState) {
|
|
|
|
this.log("WindowCovering PositionState characteristic enabled");
|
|
|
|
this.log("["+ this.name +"]:WindowCovering PositionState characteristic enabled");
|
|
|
|
this.bindCharacteristic(myService, Characteristic.PositionState, "Float", config.PositionState);
|
|
|
|
this.bindCharacteristic(myService, Characteristic.PositionState, "Float", config.PositionState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return myService;
|
|
|
|
return myService;
|
|
|
|
@@ -910,7 +961,7 @@ KNXDevice.prototype = {
|
|
|
|
informationService
|
|
|
|
informationService
|
|
|
|
.setCharacteristic(Characteristic.Manufacturer, "Opensource Community")
|
|
|
|
.setCharacteristic(Characteristic.Manufacturer, "Opensource Community")
|
|
|
|
.setCharacteristic(Characteristic.Model, "KNX Universal Device")
|
|
|
|
.setCharacteristic(Characteristic.Model, "KNX Universal Device")
|
|
|
|
.setCharacteristic(Characteristic.SerialNumber, "Version 1.1.2");
|
|
|
|
.setCharacteristic(Characteristic.SerialNumber, "Version 1.1.4");
|
|
|
|
|
|
|
|
|
|
|
|
accessoryServices.push(informationService);
|
|
|
|
accessoryServices.push(informationService);
|
|
|
|
|
|
|
|
|
|
|
|
@@ -965,8 +1016,8 @@ KNXDevice.prototype = {
|
|
|
|
accessoryServices.push(this.getWindowCoveringService(configService));
|
|
|
|
accessoryServices.push(this.getWindowCoveringService(configService));
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
default:
|
|
|
|
this.log("[ERROR] unknown 'type' property of '"+configService.type+"' for service "+ configService.name + " in config.json. KNX platform section fault ");
|
|
|
|
this.log("[ERROR] unknown 'type' property of ["+configService.type+"] for service ["+ configService.name + "] in config.json. KNX platform section fault ");
|
|
|
|
//throw new Error("[ERROR] unknown 'type' property for service "+ configService.name + " in config.json. KNX platform section fault ");
|
|
|
|
throw new Error("[ERROR] unknown 'type' property of ["+configService.type+"] for service '"+ configService.name + "' in config.json. KNX platform section fault ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// start listening for events on the bus (if not started yet - will prevent itself)
|
|
|
|
// start listening for events on the bus (if not started yet - will prevent itself)
|
|
|
|
|