Files
homebridge/platforms/ZWayServer.js
S'pht'Kr f72cb43043 Numerous fixes and improvements.
Committing before merging back to master, since the upstream branches have merged into master!
2015-08-24 06:48:34 +02:00

666 lines
24 KiB
JavaScript

var types = require("HAP-NodeJS/accessories/types.js");
var request = require("request");
var tough = require('tough-cookie');
var Q = require("q");
var zwshkDeviceClasses = [
{
primaryType: "switchBinary",
subTypes: {
"battery": true,
"sensorMultilevel.Electric": true
},
tcType: types.SWITCH_TCTYPE
}
,
{
primaryType: "thermostat",
subTypes: {
"sensorMultiLevel.Temperature": true,
"battery": true
},
tcType: types.THERMOSTAT_TCTYPE
}
,
{
primaryType: "sensorBinary.Door/Window",
subTypes: {
"battery": true
},
tcType: types.SENSOR_TCTYPE
}
,
{
primaryType: "sensorMultilevel.Temperature",
subTypes: {
"battery": true
},
tcType: types.SENSOR_TCTYPE
}
,
{
primaryType: "switchMultilevel",
subTypes: {
"battery": true
},
tcType: types.LIGHTBULB_TCTYPE
}
];
function ZWayServerPlatform(log, config){
this.log = log;
this.url = config["url"];
this.login = config["login"];
this.password = config["password"];
this.name_overrides = config["name_overrides"];
this.batteryLow = config["battery_low_level"];
this.userAgent = "HomeBridge/-1^0.5";
this.sessionId = "";
this.jar = request.jar(new tough.CookieJar());
}
ZWayServerPlatform.getVDevTypeKey = function(vdev){
return vdev.deviceType + (vdev.metrics && vdev.metrics.probeTitle ? "." + vdev.metrics.probeTitle : "")
}
ZWayServerPlatform.getVDevServiceTypes = function(vdev){
var typeKey = ZWayServerPlatform.getVDevTypeKey(vdev);
switch (typeKey) {
case "switchBinary":
return [types.SWITCH_STYPE];
case "switchMultilevel":
return [types.LIGHTBULB_STYPE];
case "thermostat":
return [types.THERMOSTAT_STYPE];
case "sensorMultilevel.Temperature":
return [types.TEMPERATURE_SENSOR_STYPE];
case "sensorBinary.Door/Window":
return [types.GARAGE_DOOR_OPENER_STYPE];
}
}
ZWayServerPlatform.getVDevCharacteristicsTypes = function(vdev){
var typeKey = ZWayServerPlatform.getVDevTypeKey(vdev);
switch (typeKey) {
case "switchBinary":
return [types.POWER_STATE_CTYPE];
case "switchMultilevel":
return [types.POWER_STATE_CTYPE, types.BRIGHTNESS_CTYPE];
case "thermostat":
return [types.TARGET_TEMPERATURE_CTYPE, types.TEMPERATURE_UNITS_CTYPE, types.CURRENTHEATINGCOOLING_CTYPE, types.TARGETHEATINGCOOLING_CTYPE];
case "sensorMultilevel.Temperature":
return [types.CURRENT_TEMPERATURE_CTYPE, types.TEMPERATURE_UNITS_CTYPE];
case "sensorBinary.Door/Window":
return [types.CURRENT_DOOR_STATE_CTYPE, types.TARGET_DOORSTATE_CTYPE, types.OBSTRUCTION_DETECTED_CTYPE];
case "battery.Battery":
return [types.BATTERY_LEVEL_CTYPE, types.STATUS_LOW_BATTERY_CTYPE];
}
}
ZWayServerPlatform.prototype = {
zwayRequest: function(opts){
var that = this;
var deferred = Q.defer();
opts.jar = true;//this.jar;
opts.json = true;
opts.headers = {
"Cookie": "ZWAYSession=" + this.sessionId
};
//opts.proxy = 'http://localhost:8888';
request(opts, function(error, response, body){
if(response.statusCode == 401){
that.log("Authenticating...");
request({
method: "POST",
url: that.url + 'ZAutomation/api/v1/login',
//proxy: 'http://localhost:8888',
body: { //JSON.stringify({
"form": true,
"login": that.login,
"password": that.password,
"keepme": false,
"default_ui": 1
},
headers: {
"Accept": "application/json",
"Content-Type": "application/json",
"User-Agent": that.userAgent
},
json: true,
jar: true//that.jar
}, function(error, response, body){
if(response.statusCode == 200){
that.sessionId = body.data.sid;
opts.headers["Cookie"] = "ZWAYSession=" + that.sessionId;
that.log("Authenticated. Resubmitting original request...");
request(opts, function(error, response, body){
if(response.statusCode == 200){
deferred.resolve(body);
} else {
deferred.reject(response);
}
});
} else {
deferred.reject(response);
}
});
} else if(response.statusCode == 200) {
deferred.resolve(body);
} else {
deferred.reject(response);
}
});
return deferred.promise;
}
,
accessories: function(callback) {
this.log("Fetching Z-Way devices...");
var that = this;
var foundAccessories = [];
this.zwayRequest({
method: "GET",
url: this.url + 'ZAutomation/api/v1/devices'
}).then(function(result){
var devices = result.data.devices;
var groupedDevices = {};
for(var i = 0; i < devices.length; i++){
var vdev = devices[i];
if(vdev.tags.indexOf("HomeBridge:Skip") >= 0) { that.log("Tag says skip!"); continue; }
var gdid = vdev.id.replace(/^(.*?)_zway_(\d+-\d+)-\d.*/, '$1_$2');
var gd = groupedDevices[gdid] || (groupedDevices[gdid] = {devices: [], types: {}, primary: undefined});
gd.devices.push(vdev);
gd.types[ZWayServerPlatform.getVDevTypeKey(vdev)] = gd.devices.length - 1;
gd.types[vdev.deviceType] = gd.devices.length - 1; // also include the deviceType only as a possibility
}
//TODO: Make a second pass, re-splitting any devices that don't make sense together
for(var gdid in groupedDevices) {
if(!groupedDevices.hasOwnProperty(gdid)) continue;
// Debug/log...
that.log('Got grouped device ' + gdid + ' consiting of devices:');
var gd = groupedDevices[gdid];
for(var j = 0; j < gd.devices.length; j++){
that.log(gd.devices[j].id + " - " + gd.devices[j].deviceType + (gd.devices[j].metrics && gd.devices[j].metrics.probeTitle ? "." + gd.devices[j].metrics.probeTitle : ""));
}
var accessory = null;
for(var ti = 0; ti < zwshkDeviceClasses.length; ti++){
if(gd.types[zwshkDeviceClasses[ti].primaryType] !== undefined){
gd.primary = gd.types[zwshkDeviceClasses[ti].primaryType];
var pd = gd.devices[gd.primary];
var name = pd.metrics && pd.metrics.title ? pd.metrics.title : pd.id;
that.log("Using class with primaryType " + zwshkDeviceClasses[ti].primaryType + ", " + name + " (" + pd.id + ") as primary.");
accessory = new ZWayServerAccessory(name, zwshkDeviceClasses[ti], gd, that);
break;
}
}
if(!accessory)
that.log("WARN: Didn't find suitable device class!");
else
foundAccessories.push(accessory);
}
//foundAccessories = foundAccessories.slice(0, 10); // Limit to a few devices for testing...
callback(foundAccessories);
});
}
}
function ZWayServerAccessory(name, dclass, devDesc, platform) {
// device info
this.name = name;
this.dclass = dclass;
this.devDesc = devDesc;
this.platform = platform;
this.log = platform.log;
}
ZWayServerAccessory.prototype = {
command: function(vdev, command, value) {
return this.platform.zwayRequest({
method: "GET",
url: this.platform.url + 'ZAutomation/api/v1/devices/' + vdev.id + '/command/' + command,
qs: (value === undefined ? undefined : value)
});
},
informationCharacteristics: function() {
return [
{
cType: types.NAME_CTYPE,
onUpdate: null,
perms: ["pr"],
format: "string",
initialValue: this.name,
supportEvents: false,
supportBonjour: false,
manfDescription: "Name of the accessory",
designedMaxLength: 255
},{
cType: types.MANUFACTURER_CTYPE,
onUpdate: null,
perms: ["pr"],
format: "string",
initialValue: "Z-Wave.me",
supportEvents: false,
supportBonjour: false,
manfDescription: "Manufacturer",
designedMaxLength: 255
},{
cType: types.MODEL_CTYPE,
onUpdate: null,
perms: ["pr"],
format: "string",
initialValue: "VDev",
supportEvents: false,
supportBonjour: false,
manfDescription: "Model",
designedMaxLength: 255
},{
cType: types.SERIAL_NUMBER_CTYPE,
onUpdate: null,
perms: ["pr"],
format: "string",
initialValue: "VDev-" + this.devDesc.devices[this.devDesc.primary].h, //TODO: Is this valid?
supportEvents: false,
supportBonjour: false,
manfDescription: "SN",
designedMaxLength: 255
},{
cType: types.IDENTIFY_CTYPE,
onUpdate: null,
perms: ["pw"],
format: "bool",
initialValue: false,
supportEvents: false,
supportBonjour: false,
manfDescription: "Identify Accessory",
designedMaxLength: 1
}
]
},
controlCharacteristics: function(vdev) {
var that = this;
var cTypes = [];
var cxs = ZWayServerPlatform.getVDevCharacteristicsTypes(vdev);
if(!cxs || cxs.length <= 0) return cTypes;
if (cxs.indexOf(types.POWER_STATE_CTYPE) >= 0) {
cTypes.push({
cType: types.POWER_STATE_CTYPE,
onUpdate: function(value) {
if (value == 0) {
that.command(vdev, "off");
} else {
that.command(vdev, "on");
}
},
perms: ["pw","pr","ev"],
format: "bool",
initialValue: 0,
supportEvents: true,
supportBonjour: false,
manfDescription: "Change the power state",
designedMaxLength: 1
});
}
if (cxs.indexOf(types.BRIGHTNESS_CTYPE) >= 0) {
cTypes.push({
cType: types.BRIGHTNESS_CTYPE,
onUpdate: function(value) {
that.command(vdev, "exact", {level: parseInt(value, 10)});
},
perms: ["pw","pr","ev"],
format: "int",
initialValue: 0,
supportEvents: true,
supportBonjour: false,
manfDescription: "Adjust Brightness of Light",
designedMinValue: 0,
designedMaxValue: 100,
designedMinStep: 1,
unit: "%"
});
}
if (cxs.indexOf(types.CURRENT_TEMPERATURE_CTYPE) >= 0) {
cTypes.push({
cType: types.CURRENT_TEMPERATURE_CTYPE,
onUpdate: null,
onRead: function(callback) {
that.platform.zwayRequest({
method: "GET",
url: that.platform.url + 'ZAutomation/api/v1/devices/' + vdev.id
}).then(function(result){
callback(result.data.metrics.level);
});
},
perms: ["pr","ev"],
format: "int",
initialValue: 20,
supportEvents: false,
supportBonjour: false,
manfDescription: "Current Temperature",
unit: "celsius"
});
}
if (cxs.indexOf(types.TARGET_TEMPERATURE_CTYPE) >= 0) {
cTypes.push({
cType: types.TARGET_TEMPERATURE_CTYPE,
onUpdate: function(value) {
try {
that.command(vdev, "exact", {level: parseFloat(value)});
}
catch (e) {
that.log(e);
}
},
onRead: function(callback) {
that.platform.zwayRequest({
method: "GET",
url: that.platform.url + 'ZAutomation/api/v1/devices/' + vdev.id
}).then(function(result){
callback(result.data.metrics.level);
});
},
perms: ["pw","pr","ev"],
format: "int",
initialValue: 20,
supportEvents: false,
supportBonjour: false,
manfDescription: "Target Temperature",
designedMinValue: vdev.metrics && vdev.metrics.min !== undefined ? vdev.metrics.min : 5,
designedMaxValue: vdev.metrics && vdev.metrics.max !== undefined ? vdev.metrics.max : 40,
designedMinStep: 1,
unit: "celsius"
});
}
if (cxs.indexOf(types.TEMPERATURE_UNITS_CTYPE) >= 0) {
cTypes.push({
cType: types.TEMPERATURE_UNITS_CTYPE,
perms: ["pr"],
format: "int",
//TODO: Let this update from the device if it changes after startup.
initialValue: vdev.metrics.scaleTitle.indexOf("F") >= 0 ? 1 : 0,
supportEvents: false,
supportBonjour: false,
manfDescription: "Unit",
});
}
if (cxs.indexOf(types.CURRENTHEATINGCOOLING_CTYPE) >= 0) {
cTypes.push({
cType: types.CURRENTHEATINGCOOLING_CTYPE,
//TODO: Support multifunction thermostats...only heating supported now.
/*
onUpdate: null,
onRead: function(callback) {
that.getCurrentHeatingCooling(function(currentHeatingCooling){
callback(currentHeatingCooling);
});
},
*/
perms: ["pr"],
format: "int",
initialValue: 1,
supportEvents: false,
supportBonjour: false,
manfDescription: "Current Mode",
designedMaxLength: 1,
designedMinValue: 0,
designedMaxValue: 2,
designedMinStep: 1,
});
}
if (cxs.indexOf(types.TARGETHEATINGCOOLING_CTYPE) >= 0) {
cTypes.push({
cType: types.TARGETHEATINGCOOLING_CTYPE,
//TODO: Support multifunction thermostats...only heating supported now.
/*
onUpdate: function(value) {
that.setTargetHeatingCooling(value);
},
onRead: function(callback) {
that.getTargetHeatingCoooling(function(targetHeatingCooling){
callback(targetHeatingCooling);
});
},
*/
perms: ["pr"],
format: "int",
initialValue: 0,
supportEvents: false,
supportBonjour: false,
manfDescription: "Target Mode",
designedMinValue: 0,
designedMaxValue: 3,
designedMinStep: 1,
});
}
if (cxs.indexOf(types.CONTACT_SENSOR_STATE_CTYPE) >= 0) {
cTypes.push({
cType: types.CONTACT_SENSOR_STATE_CTYPE,
onUpdate: null,
onRead: function(callback) {
that.platform.zwayRequest({
method: "GET",
url: that.platform.url + 'ZAutomation/api/v1/devices/' + vdev.id
}).then(function(result){
callback(result.data.metrics.level == "off" ? 1 : 0);
});
},
perms: ["pr","ev"],
format: "bool",
initialValue: 1,
supportEvents: false,
supportBonjour: false,
manfDescription: "Contact State",
designedMaxLength: 1
});
}
if (cxs.indexOf(types.CURRENT_DOOR_STATE_CTYPE) >= 0) {
cTypes.push({
cType: types.CURRENT_DOOR_STATE_CTYPE,
onRead: function(callback) {
that.platform.zwayRequest({
method: "GET",
url: that.platform.url + 'ZAutomation/api/v1/devices/' + vdev.id
}).then(function(result){
callback(result.data.metrics.level == "off" ? 1 : 0);
});
},
perms: ["pr","ev"],
format: "int",
initialValue: 1,
supportEvents: false,
supportBonjour: false,
manfDescription: "Current Door State",
designedMinValue: 0,
designedMaxValue: 4,
designedMinStep: 1,
designedMaxLength: 1
});
}
if (cxs.indexOf(types.TARGET_DOORSTATE_CTYPE) >= 0) {
cTypes.push({
cType: types.TARGET_DOORSTATE_CTYPE,
onRead: function(callback) {
that.platform.zwayRequest({
method: "GET",
url: that.platform.url + 'ZAutomation/api/v1/devices/' + vdev.id
}).then(function(result){
callback(result.data.metrics.level == "off" ? 0 : 1);
});
},
perms: ["pr","ev"], //TODO: If we support some non-sensor device that can actually open, add "pw"!
format: "int",
initialValue: 0,
supportEvents: false,
supportBonjour: false,
manfDescription: "Target Door State",
designedMinValue: 0,
designedMaxValue: 1,
designedMinStep: 1,
designedMaxLength: 1
});
}
if (cxs.indexOf(types.OBSTRUCTION_DETECTED_CTYPE) >= 0) {
cTypes.push({
cType: types.OBSTRUCTION_DETECTED_CTYPE,
perms: ["pr","ev"],
format: "bool",
initialValue: false,
supportEvents: false,
supportBonjour: false,
manfDescription: "Obstruction Detected"
});
}
if (cxs.indexOf(types.BATTERY_LEVEL_CTYPE) >= 0) {
cTypes.push({
cType: types.BATTERY_LEVEL_CTYPE,
onRead: function(callback) {
that.platform.zwayRequest({
method: "GET",
url: that.platform.url + 'ZAutomation/api/v1/devices/' + vdev.id
}).then(function(result){
callback(result.data.metrics.level);
});
},
perms: ["pr","ev"],
format: "int",
initialValue: 100,
supportEvents: true,
supportBonjour: false,
manfDescription: "Battery Level",
designedMinValue: 0,
designedMaxValue: 100,
designedMinStep: 1,
unit: "%"
});
}
if (cxs.indexOf(types.STATUS_LOW_BATTERY_CTYPE) >= 0) {
cTypes.push({
cType: types.STATUS_LOW_BATTERY_CTYPE,
onUpdate: null,
onRead: function(callback) {
that.platform.zwayRequest({
method: "GET",
url: that.platform.url + 'ZAutomation/api/v1/devices/' + vdev.id
}).then(function(result){
callback(result.data.metrics.level <= that.platform.batteryLow ? 1 : 0);
});
},
perms: ["pr","ev"],
format: "bool",
initialValue: 0,
supportEvents: false,
supportBonjour: false,
manfDescription: "Battery is low",
designedMaxLength: 1
});
}
return cTypes;
},
getServices: function() {
var that = this;
var services = [{
sType: types.ACCESSORY_INFORMATION_STYPE,
characteristics: this.informationCharacteristics(),
}];
// rearrange the array so the primary is first
var vdevs = this.devDesc.devices.concat();
var p = vdevs.splice(this.devDesc.primary, 1)[0];
vdevs.unshift(p);
/*
for(var i = 0; i < vdevs.length; i++){
var sTypes = ZWayServerPlatform.getVDevServiceTypes(vdevs[i]);
if(!sTypes) continue;
for(var j = 0; j < sTypes.length; j++){
services.push({
sType: sTypes[j],
characteristics: this.controlCharacteristics(vdevs[i])
});
}
}
*/
var sTypes = ZWayServerPlatform.getVDevServiceTypes(vdevs[0]);
var cTypes = [{
cType: types.NAME_CTYPE,
onUpdate: null,
perms: ["pr"],
format: "string",
initialValue: this.name,
supportEvents: true,
supportBonjour: false,
manfDescription: "Name of service",
designedMaxLength: 255
}];
if(sTypes) for(var i = 0; i < vdevs.length; i++){
cTypes = cTypes.concat(this.controlCharacteristics(vdevs[i]));
}
// Scrub/eliminate duplicate cTypes? This is a lot of guesswork ATM...
var hits = {};
for (var i = 0; i < cTypes.length; i++){
if(hits[cTypes[i].cType]) cTypes.splice(i--, 1); // Remember postfix means post-evaluate!
else hits[cTypes[i].cType] = cTypes[i];
}
// Thermostats MUST include current temperature...so, for the Danfoss/Devolo radiator
// thermostats, we have to fake one...
if (hits[types.TARGET_TEMPERATURE_CTYPE] && !hits[types.CURRENT_TEMPERATURE_CTYPE]) {
// Copy the "target" device to the "current" one, with necessary tweaks...
var tcx = hits[types.TARGET_TEMPERATURE_CTYPE];
var ccx = {};
for(var p in tcx){
if(tcx.hasOwnProperty(p)) ccx[p] = tcx[p];
}
ccx.cType = types.CURRENT_TEMPERATURE_CTYPE;
ccx.onUpdate = null;
ccx.perms = ["pr"];
//ccx.onRead = null; // Override this??
cTypes.push(ccx);
}
services.push({
sType: sTypes[0],
characteristics: cTypes
});
//...
this.log("Loaded services for " + this.name);
return services;
}
};
module.exports.accessory = ZWayServerAccessory;
module.exports.platform = ZWayServerPlatform;