Monday, 10 April 2017

Home Automation - HomeAssistant and Clipsal C-Bus

This was an interesting one, I wanted to control C-Bus lights / Devices in the home with HomeAssistant, it took a bit to get going, but worth it..

This isn't a complete guide on HomeAssistant, MQTT, Mosquitto, C-Gate, Debian or anything else, but with the help of Google and luck, it should get you going..

Components used for the interface:
Lots of reading the C-Gate programming guide and other places:
www3.clipsal.com/cis/downloads/Toolkit/CGateServerGuide_1_0.pdf

http://addictedtopi.tumblr.com/post/96351714013/installing-c-gate-on-a-raspberry-pi


Setting up Debian:
Install other packages openjdk-7-jdk nodejs npm mosquitto mosquitto-clients

Setting up Node.js
npm install mqtt --save

C-Bus / C-Gate setup..
Install C-Gate on linux and connect to this remote C-Gate with the C-Bus toolkit, get everything working with this configuration BEFORE continuing..  The rest won't really work until you have networks / projects etc set up.  The configuration is stored on the linux server with the C-Gate install.

I used this post as a guide to install:
http://addictedtopi.tumblr.com/post/96351714013/installing-c-gate-on-a-raspberry-pi









Modifications to the code required..
I assume the excellent C-Gate / MQTT code from the1las @ github was written for an older version of C-Gate than the one I'm using (2.10.6) so I had to make some modifications to get it to work for me..  Without this code as a base, this would have taken so much longer!

modified index.js

var mqtt = require('mqtt'), url = require('url');
var net = require('net');
var events = require('events');
var settings = require('./settings.js');

var buffer = "";
var eventEmitter = new events.EventEmitter();

// MQTT URL
var mqtt_url = url.parse('mqtt://'+settings.mqtt);

// Username and password
var OPTIONS = {};
if(settings.mqttusername && settings.mqttpassword) {
  OPTIONS.username = settings.mqttusername;
  OPTIONS.password = settings.mqttpassword;
}

// Create an MQTT client connection
var client = mqtt.createClient(mqtt_url.port, mqtt_url.hostname,OPTIONS);

var HOST = settings.cbusip;
var COMPORT = 20023;
var EVENTPORT = 20025;

var logging = settings.logging;

// Connect to cgate via telnet
var command = new net.Socket();
command.connect(COMPORT, HOST, function() {

  console.log('CONNECTED TO C-GATE COMMAND PORT: ' + HOST + ':' + COMPORT);
  command.write('EVENT ON\n');

});


// Connect to cgate event port via telnet
var event = new net.Socket();
event.connect(EVENTPORT, HOST, function() {
  command.write('PROJECT LOAD '+settings.cbusname+'\n');
  command.write('PROJECT USE '+settings.cbusname+'\n');
  command.write('NET OPEN '+settings.cbusnetwork+'\n');
  command.write('PROJECT START'+'\n');

  console.log('CONNECTED TO C-GATE EVENT PORT: ' + HOST + ':' + EVENTPORT);

});


client.on('connect', function() { // When connected
  console.log('CONNECTED TO MQTT: ' + settings.mqtt);

  // Subscribe to MQTT
  client.subscribe('cbus/write/#', function() {

    // when a message arrives, do something with it
    client.on('message', function(topic, message, packet) {
      if (logging == true) {console.log('Message received on ' + topic + ' : ' + message);}

      parts = topic.split("/");
      if (parts.length > 5)

      switch(parts[5].toLowerCase()) {

        // Get updates from all groups
        case "getall":
          command.write('GET /'+parts[2]+'/'+parts[3]+'/* level\n');
          break;

        // On/Off control
        case "switch":

          if(message == "ON") {command.write('ON '+parts[2]+'/'+parts[3]+'/'+parts[4]+'\n')};
          if(message == "OFF") {command.write('OFF '+parts[2]+'/'+parts[3]+'/'+parts[4]+'\n')};
          break;

        // Ramp, increase/decrease, on/off control
        case "ramp":
          switch(message.toUpperCase()) {
            case "INCREASE":
              eventEmitter.on('level',function increaseLevel(address,level) {
                if (address == parts[2]+'/'+parts[3]+'/'+parts[4]) {
                  command.write('RAMP '+parts[2]+'/'+parts[3]+'/'+parts[4]+' '+Math.min((level+26),255)+' '+'\n');
                  eventEmitter.removeListener('level',increaseLevel);
                }
              });
              command.write('GET '+parts[2]+'/'+parts[3]+'/'+parts[4]+' level\n');

              break;

            case "DECREASE":
              eventEmitter.on('level',function decreaseLevel(address,level) {
                if (address == parts[2]+'/'+parts[3]+'/'+parts[4]) {
                  command.write('RAMP '+parts[2]+'/'+parts[3]+'/'+parts[4]+' '+Math.max((level-26),0)+' '+'\n');
                  eventEmitter.removeListener('level',decreaseLevel);
                }
              });
              command.write('GET '+parts[2]+'/'+parts[3]+'/'+parts[4]+' level\n');

              break;

            case "ON":
              command.write('ON '+parts[2]+'/'+parts[3]+'/'+parts[4]+'\n');
              break;
            case "OFF":
              command.write('OFF '+parts[2]+'/'+parts[3]+'/'+parts[4]+'\n');
              break;
            default:
              var ramp = message.split(",");
              var num = Math.round(parseInt(ramp[0])*255/100)
              if (!isNaN(num) && num < 256) {

                if (ramp.length > 1) {
                  command.write('RAMP '+parts[2]+'/'+parts[3]+'/'+parts[4]+' '+num+' '+ramp[1]+'\n');
                } else {
                  command.write('RAMP '+parts[2]+'/'+parts[3]+'/'+parts[4]+' '+num+'\n');
                }
              }
          }
          break;
        default:
      }
    });
  });

  // publish a message to a topic
  client.publish('hello/world', 'CBUS ON', function() {
  });
});



command.on('data',function(data) {
  var lines = (buffer+data.toString()).split("\n");
  buffer = lines[lines.length-1];
  if (lines.length > 1) {
    for (i = 0;i 1 && parts1[0] == "300") {
        var parts2 = parts1[1].toString().split(" ");

        address = (parts2[0].substring(0,parts2[0].length-1)).split("/");
        var level = parts2[1].split("=");
        if (parseInt(level[1]) == 0) {
          if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' OFF');}
          if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' 0%');}
          client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/state' , 'OFF' );
          client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/level' , '0' );
          eventEmitter.emit('level',address[3]+'/'+address[4]+'/'+address[5],0);
        } else {
          if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' ON');}
          if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' '+Math.round(parseInt(level[1])*100/255).toString()+'%');}
          client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/state' , 'ON' );
          client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/level' , Math.round(parseInt(level[1])*100/255).toString() );
          eventEmitter.emit('level',address[3]+'/'+address[4]+'/'+address[5],Math.round(parseInt(level[1])));

        }
      } else {
        var parts2 = parts1[0].toString().split(" ");
        if (parts2[0] == "300") {
        address = (parts2[1].substring(0,parts2[1].length-1)).split("/");
        var level = parts2[2].split("=");
        if (parseInt(level[1]) == 0) {
          if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' OFF');}
          if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' 0%');}
          client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/state' , 'OFF' );
          client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/level' , '0' );
          eventEmitter.emit('level',address[3]+'/'+address[4]+'/'+address[5],0);
        } else {
          if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' ON');}
          if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' '+Math.round(parseInt(level[1])*100/255).toString()+'%');}
          client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/state' , 'ON' );
          client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/level' , Math.round(parseInt(level[1])*100/255).toString() );
          eventEmitter.emit('level',address[3]+'/'+address[4]+'/'+address[5],Math.round(parseInt(level[1])));

        }

        }
      }
    }
  }
});
// Add a 'data' event handler for the client socket // data is what the server sent to this socket event.on('data', function(data) { var parts = data.toString().split(" "); if(parts[0] == "lighting") { address = parts[2].split("/"); switch(parts[1]) { case "on": if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' ON');} if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' 100%');} client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/state' , 'ON' ); client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/level' , '100' ); break; case "off": if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' OFF');} if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' 0%');} client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/state' , 'OFF' ); client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/level' , '0' ); break; case "ramp": if(parseInt(parts[3]) > 0) { if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' ON');} if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' '+Math.round(parseInt(parts[3])*100/255).toString()+'%');} client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/status' , 'ON', function() {}); client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/level' , Math.round(parseInt(parts[3])*100/255).toString(), function() {}); } else { if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' OFF');} if (logging == true) {console.log('C-Bus status received: '+address[3] +'/'+address[4]+'/'+address[5]+' 0%');} client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/status' , 'OFF' ); client.publish('cbus/read/'+address[3]+'/'+address[4]+'/'+address[5]+'/level' , '0' ); } break; default: } } });

The modifications are based on the reading of the C-Gate docs.
Basically I added some stuff on connect to C-Gate..  e.g. the lines:

  • PROJECT LOAD
  • PROJECT USE
  • NET OPEN
  • PROJECT START

The remainder of the modifications were around device control, Basically you now send
ON Network/Application/Group
or
OFF Network/Application/Group
to the C-Gate and it turns on the appropriate C-Bus Group.  The original code prefixed each command with the cbusname.. I also removed some of the parameters on the mqtt client.publish statements as they caused inconsistencies in my setup.

modified settings.js

//cbus ip address
exports.cbusip = '127.0.0.1';


//cbus project name
exports.cbusname = "HOME";
exports.cbusnetwork = "254";

//mqtt server ip:port
exports.mqtt = '127.0.0.1:1883';

//username and password (unncomment to use)
//exports.mqttusername = 'user1';
//exports.mqttpassword = 'password1';


//logging
exports.logging = true;

The only difference here was adding the cbus network variable.


Next was adding the MQTT config in HomeAssistant (configuration.yaml).  This points at the IP address where you installed Mosquitto..

mqtt:
  broker: 192.168.X.X
  port: 1883
  client_id: home-assistant-1
  keepalive: 60
  protocol: 3.1

This just allows MQTT to be used, it doesn't define anything to switch on / off...

I control some lights as follows (existing HUE lughts, then the C-Bus Lights):

light:
  - platform: hue
    host: 192.168.X.X
  - platform: mqtt
    name: Entrance (CBUS)
    state_topic: "cbus/read/254/56/2/state"
    command_topic: "cbus/write/254/56/2/switch"      
  - platform: mqtt
    name: Outside Front (CBUS)
    state_topic: "cbus/read/254/56/3/state"
    command_topic: "cbus/write/254/56/3/switch"  
  - platform: mqtt
    name: Lounge (CBUS)
    state_topic: "cbus/read/254/56/4/state"
    command_topic: "cbus/write/254/56/4/switch"

The command_topic / state topic contain the C-Bus Network/Application/Group being controlled

You can also add "Switches" rather than lights
switch:
  - platform: mqtt
    name: Mozzie Zapper (CBUS)
    state_topic: "cbus/read/254/56/11/state"
    command_topic: "cbus/write/254/56/11/switch"  

The control on these is 2-way, so if i turn on a C-Bus light with the wall switch, the change is reflected in HomeAssistant.

To get the Network/Application/Group list for my C-Bus setup:
Look in the C-Bus Toolkit..

The Network number is displayed at the top.  The Application number is displayed on the Application properties.
The Group Address is displayed on the Group to be controlled.


Starting all at boot:

C-Gate started as per this post
http://addictedtopi.tumblr.com/post/96351714013/installing-c-gate-on-a-raspberry-pi

The node.js code is loaded at boot on the debian box (jessie) by the forever module..
after doing an "npm install -g" to load the forever stuff for node.js, i had to create a symlink

ln -s /usr/bin/nodejs /usr/bin/node

or forever doesn't work (it looks for node, not nodejs).

Startup script /etc/init.d/nodeup

#!/bin/sh
#/etc/init.d/nodeup

export PATH=$PATH:/usr/local/bin
export NODE_PATH=$NODE_PATH:/usr/local/lib/node_modules

case "$1" in
  start)
  exec forever start --sourceDir=/usr/local/bin/nodejs -p /root/.forever/ index.js
  ;;
stop)
  exec forever stop --sourceDir=/usr/local/bin/nodejs index.js
  ;;
*)
  echo "Usage: /etc/init.d/nodeup {start|stop}"
  exit 1
  ;;
esac

exit 0

Nothing special, just start the init script at boot as per usual..


The future...
Oneday I might modify the node.js code to discover all C-Bus applications and devices, but for now I can control my lights from anywhere and life is good!

Friday, 23 March 2012

LDAP (slapd) and GOsa on Debian Squeeze

First "Solve" for the blog.


How to install Open LDAP (slapd) and GOsa on Debian Squeeze.  Sometime in recent history, the previous configuration file setup for LDAP (slapd.conf) was deprecated and a new config structure was put in place.  As as today (24/03/2012) the instructions for configuring LDAP on the debian pages still say that their instructions need updating.


So as with all debian installs, I started with the following:
apt-get install slapd gosa gosa-schema

Use your domain name, or local / internal domain name for the slapd install.
Enter whatever you like for the organisation name, or just press enter to accept the default.



Enter an admin password.  Probably good (as with all passwords) not to make it easily guessable, and not the same as other passwords on the same machine.


Pick either type for the database backend. I chose HDB after a bit of searching about the differences.


Next step is the bit that really got me stumped, there wasn't a lot of help on Google for how to do this.

First create a file called /etc/ldap/convert.conf with the following contents:


#Exisiting ldap schemas
include /etc/ldap/schema/core.schema
include /etc/ldap/schema/cosine.schema
include /etc/ldap/schema/nis.schema
include /etc/ldap/schema/inetorgperson.schema


#GOSA ldap schemas, order is important.
include /etc/ldap/schema/gosa/samba3.schema
include /etc/ldap/schema/gosa/gofon.schema
include /etc/ldap/schema/gosa/gosystem.schema
include /etc/ldap/schema/gosa/goto.schema
include /etc/ldap/schema/gosa/gosa-samba3.schema
include /etc/ldap/schema/gosa/gofax.schema
include /etc/ldap/schema/gosa/goserver.schema
include /etc/ldap/schema/gosa/goto-mime.schema
include /etc/ldap/schema/gosa/trust.schema



Then make an output directory
root@ldap:/etc/ldap# mkdir /tmp/convert_out


Then run the convert tool as follows:

root@ldap:/etc/ldap# slaptest -f convert.conf -F /tmp/convert_out
config file testing succeeded


This will have created some files in /tmp/convert_out.  We only want the schema files from this output, and we want them to end up in the directories for the new configuration files for slapd.  We also need to change the owner / group on these files so the slapd process can read them.

cp -p /tmp/convert_out/cn\=config/cn\=schema* /etc/ldap/slapd.d/cn\=config/
chown -R openldap.openldap /etc/ldap/slapd.d/ 

At this point, I restarted slapd, but I'm fairly certain this isn't required (your choice).

Next we get into the GOsa configuration...


Assuming Apache etc installed correctly on the machine, you should be able to go to
http://yourip/gosa/ or http://yourhostname/gosa
and see the GOsa administration pages.


At the bottom of the page, there's an "echo" command that you need to cut and paste into a terminal window.  It won't go past this step until you do.

Pick A Language


Hopefully you're all green here.  Mine was a fresh install on a clean debian install, so it looks like it should work out of the box.


On this screen you should see the "succeeded" text at the bottom, If not, something else is wrong, try re-starting slapd.


On this screen, enter "cn=admin" in the Admin DN field, and type the Admin password that you used during the LDAP install.  Then tick the auto append checkbox and click next.


Assuming you've done everything right, you should see this screen...

And that's where i'll end this post, after this the documentation all around is pretty good for GOsa and LDAP. 

Enjoy.   As always feedback / suggestions are welcome.

Why?

My most recent "solve" was last week, and it wasn't until I figured it out that I thought I should probably start writing about this kind of thing.

So this blog will be my place to dump instructions, solutions, screen shots and links for the crazy list of things I solve after finding no (or little) help searching the web.