# $Id: 30_HUEBridge.pm 8979 2015-07-15 19:30:30Z justme1968 $ # "Hue Personal Wireless Lighting" is a trademark owned by Koninklijke Philips Electronics N.V., # see www.meethue.com for more information. # I am in no way affiliated with the Philips organization. package main; use strict; use warnings; use POSIX; use JSON; use Data::Dumper; use HttpUtils; my $HUEBridge_isFritzBox = undef; sub HUEBridge_isFritzBox() { $HUEBridge_isFritzBox = int( qx( [ -f /usr/bin/ctlmgr_ctl ] && echo 1 || echo 0 ) ) if( !defined( $HUEBridge_isFritzBox) ); return $HUEBridge_isFritzBox; } sub HUEBridge_Initialize($) { my ($hash) = @_; # Provider $hash->{ReadFn} = "HUEBridge_Read"; $hash->{WriteFn} = "HUEBridge_Read"; $hash->{Clients} = ":HUEDevice:"; #Consumer $hash->{DefFn} = "HUEBridge_Define"; $hash->{NotifyFn} = "HUEBridge_Notify"; $hash->{SetFn} = "HUEBridge_Set"; $hash->{GetFn} = "HUEBridge_Get"; $hash->{UndefFn} = "HUEBridge_Undefine"; $hash->{AttrList}= "key disable:1 httpUtils:1,0 pollDevices:1 queryAfterSet:1"; } sub HUEBridge_Read($@) { my ($hash,$chash,$name,$id,$obj)= @_; if( $id =~ m/^G(\d.*)/ ) { return HUEBridge_Call($hash, $chash, 'groups/' . $1, $obj); } elsif( $id =~ m/^S(\d.*)/ ) { return HUEBridge_Call($hash, $chash, 'sensors/' . $1, $obj); } return HUEBridge_Call($hash, $chash, 'lights/' . $id, $obj); } sub HUEBridge_Detect($) { my ($hash) = @_; my $name = $hash->{NAME}; Log3 $name, 3, "HUEBridge_Detect"; my ($err,$ret) = HttpUtils_BlockingGet({ url => "http://www.meethue.com/api/nupnp", method => "GET", }); if( defined($err) && $err ) { Log3 $name, 3, "HUEBridge_Detect: error detecting bridge: ".$err; return; } my $host = ''; if( defined($ret) && $ret ne '' && $ret =~ m/^[\[{].*[\]}]$/ ) { my $obj = from_json($ret); if( defined($obj->[0]) && defined($obj->[0]->{'internalipaddress'}) ) { $host = $obj->[0]->{'internalipaddress'}; } } if( !defined($host) || $host eq '' ) { Log3 $name, 3, 'HUEBridge_Detect: error detecting bridge.'; return; } Log3 $name, 3, "HUEBridge_Detect: ${host}"; $hash->{Host} = $host; return $host; } sub HUEBridge_Define($$) { my ($hash, $def) = @_; my @args = split("[ \t]+", $def); return "Usage: define HUEBridge [] [interval]" if(@args < 2); my ($name, $type, $host, $interval) = @args; if( !defined($host) ) { $hash->{NUPNP} = 1; HUEBridge_Detect($hash); } else { delete $hash->{NUPNP}; } $interval= 300 unless defined($interval); if( $interval < 10 ) { $interval = 10; } $hash->{STATE} = 'Initialized'; $hash->{Host} = $host; $hash->{INTERVAL} = $interval; $attr{$name}{"key"} = join "",map { unpack "H*", chr(rand(256)) } 1..16 unless defined( AttrVal($name, "key", undef) ); $hash->{helper}{last_config_timestamp} = 0; if( !defined($hash->{helper}{count}) ) { $modules{$hash->{TYPE}}{helper}{count} = 0 if( !defined($modules{$hash->{TYPE}}{helper}{count}) ); $hash->{helper}{count} = $modules{$hash->{TYPE}}{helper}{count}++; } $hash->{NOTIFYDEV} = "global"; if( $init_done ) { HUEBridge_OpenDev( $hash ) if( !AttrVal($name, "disable", 0) ); } return undef; } sub HUEBridge_Notify($$) { my ($hash,$dev) = @_; my $name = $hash->{NAME}; my $type = $hash->{TYPE}; return if($dev->{NAME} ne "global"); return if(!grep(m/^INITIALIZED|REREADCFG$/, @{$dev->{CHANGED}})); return undef if( AttrVal($name, "disable", 0) ); HUEBridge_OpenDev($hash); return undef; } sub HUEBridge_Undefine($$) { my ($hash,$arg) = @_; RemoveInternalTimer($hash); return undef; } sub HUEBridge_OpenDev($) { my ($hash) = @_; my $name = $hash->{NAME}; HUEBridge_Detect($hash) if( defined($hash->{NUPNP}) ); my ($err,$ret) = HttpUtils_BlockingGet({ url => "http://$hash->{Host}/description.xml", method => "GET", timeout => 3, }); if( defined($err) && $err ) { Log3 $name, 3, "HUEBridge_Detect: error reading description: ".$err; } else { $ret =~ m/([^<]*)/; $hash->{modelName} = $1; } my $result = HUEBridge_Call($hash, undef, 'config', undef); if( !defined($result) ) { return undef; } if( !defined($result->{'linkbutton'}) ) { HUEBridge_Pair($hash); return; } $hash->{mac} = $result->{'mac'}; $hash->{STATE} = 'Connected'; HUEBridge_GetUpdate($hash); HUEBridge_Autocreate($hash); return undef; } sub HUEBridge_Pair($) { my ($hash) = @_; my $name = $hash->{NAME}; $hash->{STATE} = 'Pairing'; my $result = HUEBridge_Register($hash); if( $result->{'error'} ) { RemoveInternalTimer($hash); InternalTimer(gettimeofday()+5, "HUEBridge_Pair", $hash, 0); return undef; } $attr{$name}{key} = $result->{success}{username} if( $result->{success}{username} ); $hash->{STATE} = 'Paired'; HUEBridge_OpenDev($hash); return undef; } sub HUEBridge_string2array($) { my ($lights) = @_; my %lights = (); foreach my $part ( split(',', $lights) ) { my $light = $part; $light = $defs{$light}{ID} if( defined $defs{$light} && $defs{$light}{TYPE} eq 'HUEDevice' ); if( $light =~ m/^G/ ) { my $lights = $defs{$part}->{lights}; if( $lights ) { foreach my $light ( split(',', $lights) ) { $lights{$light} = 1; } } } else { $lights{$light} = 1; } } my @lights = sort {$a<=>$b} keys(%lights); return \@lights; } sub HUEBridge_Set($@) { my ($hash, $name, $cmd, @args) = @_; my ($arg, @params) = @args; # usage check if($cmd eq 'statusRequest') { return "usage: statusRequest" if( @args != 0 ); $hash->{LOCAL} = 1; #RemoveInternalTimer($hash); HUEBridge_GetUpdate($hash); delete $hash->{LOCAL}; return undef; } elsif($cmd eq 'swupdate') { return "usage: swupdate" if( @args != 0 ); my $obj = { 'swupdate' => { 'updatestate' => 3, }, }; my $result = HUEBridge_Call($hash, undef, 'config', $obj); if( !defined($result) || $result->{'error'} ) { return $result->{'error'}->{'description'}; } $hash->{updatestate} = 3; $hash->{helper}{updatestate} = $hash->{updatestate}; $hash->{STATE} = "updating"; return "starting update"; } elsif($cmd eq 'autocreate') { return "usage: autocreate" if( @args != 0 ); return HUEBridge_Autocreate($hash,1); } elsif($cmd eq 'autodetect') { return "usage: autodetect" if( @args != 0 ); my $result = HUEBridge_Call($hash, undef, 'lights', undef, 'POST'); return $result->{success}{'/lights'} if( $result->{success} ); return $result->{error}{description} if( $result->{error} ); return undef; } elsif($cmd eq 'delete') { return "usage: delete " if( @args != 1 ); if( defined $defs{$arg} && $defs{$arg}{TYPE} eq 'HUEDevice' ) { $arg = $defs{$arg}{ID}; } return "$arg is not hue light number" if( $arg !~ m/^\d+$/ ); my $code = $name ."-". $arg; if( defined($modules{HUEDevice}{defptr}{$code}) ) { CommandDelete( undef, "$modules{HUEDevice}{defptr}{$code}{NAME}" ); CommandSave(undef,undef) if( AttrVal( "autocreate", "autosave", 1 ) ); } my $result = HUEBridge_Call($hash, undef, "lights/$arg", undef, 'DELETE'); return $result->{error}{description} if( $result->{error} ); return undef; } elsif($cmd eq 'creategroup') { return "usage: creategroup " if( @args < 2 ); my $obj = { 'name' => join( ' ', @args[0..@args-2]), 'lights' => HUEBridge_string2array($args[@args-1]), }; my $result = HUEBridge_Call($hash, undef, 'groups', $obj, 'POST'); if( $result->{success} ) { HUEBridge_Autocreate($hash); my $code = $name ."-G". $result->{success}{id}; return "created $modules{HUEDevice}{defptr}{$code}->{NAME}" if( defined($modules{HUEDevice}{defptr}{$code}) ); } return $result->{error}{description} if( $result->{error} ); return undef; } elsif($cmd eq 'deletegroup') { return "usage: deletegroup " if( @args != 1 ); if( defined $defs{$arg} && $defs{$arg}{TYPE} eq 'HUEDevice' ) { return "$arg is not a hue group" if( $defs{$arg}{ID} != m/^G/ ); $defs{$arg}{ID} =~ m/G(.*)/; $arg = $1; } my $code = $name ."-G". $arg; if( defined($modules{HUEDevice}{defptr}{$code}) ) { CommandDelete( undef, "$modules{HUEDevice}{defptr}{$code}{NAME}" ); CommandSave(undef,undef) if( AttrVal( "autocreate", "autosave", 1 ) ); } return "$arg is not hue group number" if( $arg !~ m/^\d+$/ ); my $result = HUEBridge_Call($hash, undef, "groups/$arg", undef, 'DELETE'); return $result->{error}{description} if( $result->{error} ); return undef; } elsif($cmd eq 'savescene') { return "usage: savescene " if( @args < 3 ); my $obj = { 'name' => join( ' ', @args[1..@args-2]), 'lights' => HUEBridge_string2array($args[@args-1]), }; my $result = HUEBridge_Call($hash, undef, "scenes/$arg", $obj, 'PUT'); if( $result->{success} ) { return "created $arg"; } return $result->{error}{description} if( $result->{error} ); return undef; } elsif($cmd eq 'modifyscene') { return "usage: modifyscene " if( @args < 3 ); my( $light, @aa ) = @params; $light = $defs{$light}{ID} if( defined $defs{$light} && $defs{$light}{TYPE} eq 'HUEDevice' ); my %obj; if( (my $joined = join(" ", @aa)) =~ /:/ ) { my @cmds = split(":", $joined); for( my $i = 0; $i <= $#cmds; ++$i ) { HUEDevice_SetParam(undef, \%obj, split(" ", $cmds[$i]) ); } } else { my ($cmd, $value, $value2, @a) = @aa; HUEDevice_SetParam(undef, \%obj, $cmd, $value, $value2); } my $result = HUEBridge_Call($hash, undef, "scenes/$arg/lights/$light/state", \%obj, 'PUT'); return $result->{error}{description} if( $result->{error} ); return undef; } elsif($cmd eq 'scene') { return "usage: scene " if( @args != 1 ); my $obj = { 'scene' => $arg }; my $result = HUEBridge_Call($hash, undef, "groups/0/action", $obj, 'PUT'); return $result->{error}{description} if( $result->{error} ); RemoveInternalTimer($hash); InternalTimer(gettimeofday()+10, "HUEBridge_GetUpdate", $hash, 0); return undef; } elsif($cmd eq 'deletewhitelist') { return "usage: deletewhitelist " if( @args != 1 ); my $result = HUEBridge_Call($hash, undef, "config/whitelist/$arg", undef, 'DELETE'); return $result->{error}{description} if( $result->{error} ); return undef; } elsif($cmd eq 'touchlink') { return "usage: touchlink" if( @args != 0 ); my $obj = { 'touchlink' => JSON::true }; my $result = HUEBridge_Call($hash, undef, 'config', $obj, 'PUT'); return undef if( $result->{success} ); return $result->{error}{description} if( $result->{error} ); return undef; } else { my $list = "delete creategroup deletegroup savescene modifyscene scene deletewhitelist touchlink autocreate:noArg statusRequest:noArg"; $list .= " swupdate:noArg" if( defined($hash->{updatestate}) && $hash->{updatestate} =~ '^2' ); return "Unknown argument $cmd, choose one of $list"; } } sub HUEBridge_Get($@) { my ($hash, $name, $cmd) = @_; return "$name: get needs at least one parameter" if( !defined($cmd) ); # usage check if($cmd eq 'devices' || $cmd eq 'lights') { my $result = HUEBridge_Call($hash, undef, 'lights', undef); my $ret = ""; foreach my $key ( sort {$a<=>$b} keys %{$result} ) { my $code = $name ."-". $key; my $fhem_name =""; $fhem_name = $modules{HUEDevice}{defptr}{$code}->{NAME} if( defined($modules{HUEDevice}{defptr}{$code}) ); $ret .= sprintf( "%2i: %-25s %-15s %s\n", $key, $result->{$key}{name}, $fhem_name, $result->{$key}{type} ); } $ret = sprintf( "%2s %-25s %-15s %s\n", "ID", "NAME", "FHEM", "TYPE" ) .$ret if( $ret ); return $ret; } elsif($cmd eq 'groups') { my $result = HUEBridge_Call($hash, undef, 'groups', undef); $result->{0} = { name => 'Lightset 0', type => 'LightGroup', lights => ["ALL"] }; my $ret = ""; foreach my $key ( sort {$a<=>$b} keys %{$result} ) { my $code = $name ."-G". $key; my $fhem_name =""; $fhem_name = $modules{HUEDevice}{defptr}{$code}->{NAME} if( defined($modules{HUEDevice}{defptr}{$code}) ); $result->{$key}{type} = '' if( !defined($result->{$key}{type}) ); #deCONZ fix $result->{$key}{lights} = [] if( !defined($result->{$key}{lights}) ); #deCONZ fix $ret .= sprintf( "%2i: %-15s %-15s %-15s %s\n", $key, $result->{$key}{name}, $fhem_name, $result->{$key}{type}, join( ",", @{$result->{$key}{lights}} ) ); } $ret = sprintf( "%2s %-15s %-15s %-15s %s\n", "ID", "NAME", "FHEM", "TYPE", "LIGHTS" ) .$ret if( $ret ); return $ret; } elsif($cmd eq 'scenes') { my $result = HUEBridge_Call($hash, undef, 'scenes', undef); my $ret = ""; foreach my $key ( sort {$a cmp $b} keys %{$result} ) { $ret .= sprintf( "%-20s %-20s %s\n", $key, $result->{$key}{name}, join( ",", @{$result->{$key}{lights}} ) ); } $ret = sprintf( "%-20s %-20s %s\n", "ID", "NAME", "LIGHTS" ) .$ret if( $ret ); return $ret; } elsif($cmd eq 'sensors') { my $result = HUEBridge_Call($hash, undef, 'sensors', undef); my $ret = ""; foreach my $key ( sort {$a<=>$b} keys %{$result} ) { my $code = $name ."-S". $key; my $fhem_name =""; $fhem_name = $modules{HUEDevice}{defptr}{$code}->{NAME} if( defined($modules{HUEDevice}{defptr}{$code}) ); $ret .= sprintf( "%2i: %-15s %-15s %-15s\n", $key, $result->{$key}{name}, $fhem_name, $result->{$key}{type} ); } $ret = sprintf( "%2s %-15s %-15s %-15s\n", "ID", "NAME", "FHEM", "TYPE" ) .$ret if( $ret ); return $ret; } elsif($cmd eq 'whitelist') { my $result = HUEBridge_Call($hash, undef, 'config', undef); my $ret = ""; my $whitelist = $result->{whitelist}; foreach my $key ( sort {$whitelist->{$a}{'last use date'} cmp $whitelist->{$b}{'last use date'}} keys %{$whitelist} ) { $ret .= sprintf( "%-20s %-20s %-30s %s\n", $whitelist->{$key}{'create date'}, , $whitelist->{$key}{'last use date'}, $whitelist->{$key}{name}, $key ); } $ret = sprintf( "%-20s %-20s %-30s %s\n", "CREATE", "LAST USE", "NAME", "KEY" ) .$ret if( $ret ); return $ret; } else { return "Unknown argument $cmd, choose one of devices:noArg groups:noArg scenes:noArg sensors:noArg whitelist:noArg"; } } sub HUEBridge_GetUpdate($) { my ($hash) = @_; my $name = $hash->{NAME}; if(!$hash->{LOCAL}) { RemoveInternalTimer($hash); InternalTimer(gettimeofday()+$hash->{INTERVAL}, "HUEBridge_GetUpdate", $hash, 0); } my $type; my $result; if( AttrVal($name,"pollDevices",0) ) { my ($now) = gettimeofday(); if( $hash->{LOCAL} || $now - $hash->{helper}{last_config_timestamp} > 300 ) { $result = HUEBridge_Call($hash, $hash, undef, undef); $hash->{helper}{last_config_timestamp} = $now; } else { $type = 'lights'; $result = HUEBridge_Call($hash, $hash, 'lights', undef); } } else { $type = 'config'; $result = HUEBridge_Call($hash, $hash, 'config', undef); } return undef if( !defined($result) ); HUEBridge_dispatch( {hash=>$hash,chash=>$hash,type=>$type}, undef, undef, $result ); #HUEBridge_Parse($hash, $result); return undef; } sub HUEBridge_Parse($$) { my($hash,$result) = @_; my $name = $hash->{NAME}; Log3 $name, 4, "parse status message for $name"; #Log3 $name, 5, Dumper $result; #Log 3, Dumper $result; $result = $result->{config} if( defined($result->{config}) ); $hash->{name} = $result->{name}; $hash->{swversion} = $result->{swversion}; my @l = split( '\.', $result->{apiversion} ); $hash->{helper}{apiversion} = ($l[0] << 16) + ($l[1] << 8) + $l[2]; $hash->{apiversion} = $result->{apiversion}; $hash->{zigbeechannel} = $result->{zigbeechannel}; if( defined( $result->{swupdate} ) ) { my $txt = $result->{swupdate}->{text}; readingsSingleUpdate($hash, "swupdate", $txt, 1) if( $txt && $txt ne ReadingsVal($name,"swupdate","") ); if( defined($hash->{updatestate}) ){ $hash->{STATE} = "update done" if( $result->{swupdate}->{updatestate} == 0 && $hash->{helper}{updatestate} >= 2 ); $hash->{STATE} = "update failed" if( $result->{swupdate}->{updatestate} == 2 && $hash->{helper}{updatestate} == 3 ); } $hash->{updatestate} = $result->{swupdate}->{updatestate}; $hash->{helper}{updatestate} = $hash->{updatestate}; if( $result->{swupdate}->{devicetypes} ) { my $devicetypes; $devicetypes .= 'bridge' if( $result->{swupdate}->{devicetypes}->{bridge} ); $devicetypes .= ',' if( $devicetypes && scalar(@{$result->{swupdate}->{devicetypes}->{lights}}) ); $devicetypes .= join( ",", @{$result->{swupdate}->{devicetypes}->{lights}} ) if( $result->{swupdate}->{devicetypes}->{lights} ); $hash->{updatestate} .= " [$devicetypes]" if( $devicetypes ); } } elsif ( defined( $hash->{swupdate} ) ) { delete( $hash->{updatestate} ); delete( $hash->{helper}{updatestate} ); } } sub HUEBridge_Autocreate($;$) { my ($hash,$force)= @_; my $name = $hash->{NAME}; if( !$force ) { foreach my $d (keys %defs) { next if($defs{$d}{TYPE} ne "autocreate"); return undef if(AttrVal($defs{$d}{NAME},"disable",undef)); } } my $autocreated = 0; my $result = HUEBridge_Call($hash,undef, 'lights', undef); foreach my $key ( keys %{$result} ) { my $id= $key; my $code = $name ."-". $id; if( defined($modules{HUEDevice}{defptr}{$code}) ) { Log3 $name, 5, "$name: id '$id' already defined as '$modules{HUEDevice}{defptr}{$code}->{NAME}'"; next; } my $devname = "HUEDevice" . $id; $devname = $name ."_". $devname if( $hash->{helper}{count} ); my $define= "$devname HUEDevice $id IODev=$name"; Log3 $name, 4, "$name: create new device '$devname' for address '$id'"; my $cmdret= CommandDefine(undef,$define); if($cmdret) { Log3 $name, 1, "$name: Autocreate: An error occurred while creating device for id '$id': $cmdret"; } else { $cmdret= CommandAttr(undef,"$devname alias ".$result->{$id}{name}); $cmdret= CommandAttr(undef,"$devname room HUEDevice"); $cmdret= CommandAttr(undef,"$devname IODev $name"); $autocreated++; } } $result = HUEBridge_Call($hash,undef, 'groups', undef); $result->{0} = { name => "Lightset 0", }; foreach my $key ( keys %{$result} ) { my $id= $key; my $code = $name ."-G". $id; if( defined($modules{HUEDevice}{defptr}{$code}) ) { Log3 $name, 5, "$name: id '$id' already defined as '$modules{HUEDevice}{defptr}{$code}->{NAME}'"; next; } my $devname= "HUEGroup" . $id; $devname = $name ."_". $devname if( $hash->{helper}{count} ); my $define= "$devname HUEDevice group $id IODev=$name"; Log3 $name, 4, "$name: create new group '$devname' for address '$id'"; my $cmdret= CommandDefine(undef,$define); if($cmdret) { Log3 $name, 1, "$name: Autocreate: An error occurred while creating device for id '$id': $cmdret"; } else { $cmdret= CommandAttr(undef,"$devname alias ".$result->{$id}{name}); $cmdret= CommandAttr(undef,"$devname room HUEDevice"); $cmdret= CommandAttr(undef,"$devname IODev $name"); $autocreated++; } } CommandSave(undef,undef) if( $autocreated && AttrVal( "autocreate", "autosave", 1 ) ); return "created $autocreated devices"; } sub HUEBridge_ProcessResponse($$) { my ($hash,$obj) = @_; my $name = $hash->{NAME}; #Log3 $name, 3, ref($obj); #Log3 $name, 3, "Receiving: " . Dumper $obj; if( ref($obj) eq 'ARRAY' ) { if( defined($obj->[0]->{error})) { my $error = $obj->[0]->{error}->{'description'}; $hash->{STATE} = $error; } if( !AttrVal( $name,'queryAfterSet', 0 ) ) { my $successes; my $errors; my %json = (); foreach my $item (@{$obj}) { if( my $success = $item->{success} ) { next if( ref($success) ne 'HASH' ); foreach my $key ( keys %{$success} ) { my @l = split( '/', $key ); next if( !$l[1] ); if( $l[1] eq 'lights' && $l[3] eq 'state' ) { $json{$l[2]}->{state}->{$l[4]} = $success->{$key}; $successes++; } elsif( $l[1] eq 'groups' && $l[3] eq 'action' ) { my $code = $name ."-G". $l[2]; my $d = $modules{HUEDevice}{defptr}{$code}; my $lights = $d->{lights}; foreach my $light ( split(',', $lights) ) { $json{$light}->{state}->{$l[4]} = $success->{$key}; $successes++; } } } } elsif( my $error = $item->{error} ) { my $msg = $error->{'description'}; Log3 $name, 3, $msg; $errors++; } } foreach my $id ( keys %json ) { my $code = $name ."-". $id; if( my $chash = $modules{HUEDevice}{defptr}{$code} ) { #$json{$id}->{state}->{reachable} = 1; HUEDevice_Parse( $chash, $json{$id} ); } } } #return undef if( !$errors && $successes ); return ($obj->[0]); } elsif( ref($obj) eq 'HASH' ) { return $obj; } return undef; } sub HUEBridge_Register($) { my ($hash) = @_; my $obj = { 'username' => AttrVal($hash->{NAME}, "key", ""), 'devicetype' => 'fhem', }; return HUEBridge_Call($hash, undef, undef, $obj); } #Executes a JSON RPC sub HUEBridge_Call($$$$;$) { my ($hash,$chash,$path,$obj,$method) = @_; my $name = $hash->{NAME}; #Log3 $hash->{NAME}, 5, "Sending: " . Dumper $obj; my $json = undef; $json = encode_json($obj) if $obj; # @TODO: repeat twice? for( my $attempt=0; $attempt<2; $attempt++ ) { my $blocking; my $res = undef; if( !defined($attr{$name}{httpUtils}) ) { $blocking = 1; $res = HUEBridge_HTTP_Call($hash,$path,$json,$method); } else { $blocking = $attr{$name}{httpUtils} < 1; $res = HUEBridge_HTTP_Call2($hash,$chash,$path,$json,$method); } return $res if( !$blocking || defined($res) ); Log3 $name, 3, "HUEBridge_Call: failed, retrying"; HUEBridge_Detect($hash) if( defined($hash->{NUPNP}) ); } Log3 $name, 3, "HUEBridge_Call: failed"; return undef; } #JSON RPC over HTTP sub HUEBridge_HTTP_Call($$$;$) { my ($hash,$path,$obj,$method) = @_; my $name = $hash->{NAME}; return undef if($attr{$name} && $attr{$name}{disable}); #return { state => {reachable => 0 } } if($attr{$name} && $attr{$name}{disable}); my $uri = "http://" . $hash->{Host} . "/api"; if( defined($obj) ) { $method = 'PUT' if( !$method ); if( $hash->{STATE} eq 'Pairing' ) { $method = 'POST'; } else { $uri .= "/" . AttrVal($name, "key", ""); } } else { $uri .= "/" . AttrVal($name, "key", ""); } $method = 'GET' if( !$method ); if( defined $path) { $uri .= "/" . $path; } #Log3 $name, 3, "Url: " . $uri; Log3 $name, 4, "using HUEBridge_HTTP_Request: $method ". ($path?$path:''); my $ret = HUEBridge_HTTP_Request(0,$uri,$method,undef,$obj,undef); #Log3 $name, 3, Dumper $ret; if( !defined($ret) ) { return undef; } elsif($ret eq '') { return undef; } elsif($ret =~ /^error:(\d){3}$/) { my %result = { error => "HTTP Error Code $1" }; return \%result; } if( !$ret ) { Log3 $name, 2, "$name: empty answer received for $uri"; return undef; } elsif( $ret !~ m/^[\[{].*[\]}]$/ ) { Log3 $name, 2, "$name: invalid json detected for $uri: ". Dumper $ret; return undef; } # try { # from_json($ret); # } catch { # return undef; # } return HUEBridge_ProcessResponse($hash,decode_json($ret)) if( HUEBridge_isFritzBox() ); return HUEBridge_ProcessResponse($hash,from_json($ret)); } sub HUEBridge_HTTP_Call2($$$$;$) { my ($hash,$chash,$path,$obj,$method) = @_; my $name = $hash->{NAME}; return undef if($attr{$name} && $attr{$name}{disable}); #return { state => {reachable => 0 } } if($attr{$name} && $attr{$name}{disable}); my $url = "http://" . $hash->{Host} . "/api"; my $blocking = $attr{$name}{httpUtils} < 1; $blocking = 1 if( !defined($chash) ); if( defined($obj) ) { $method = 'PUT' if( !$method ); if( $hash->{STATE} eq 'Pairing' ) { $method = 'POST'; $blocking = 1; } else { $url .= "/" . AttrVal($name, "key", ""); } } else { $url .= "/" . AttrVal($name, "key", ""); } $method = 'GET' if( !$method ); if( defined $path) { $url .= "/" . $path; } #Log3 $name, 3, "Url: " . $url; #Log 2, $path; if( $blocking ) { Log3 $name, 4, "using HttpUtils_BlockingGet: $method ". ($path?$path:''); my($err,$data) = HttpUtils_BlockingGet({ url => $url, timeout => 4, method => $method, noshutdown => 1, header => "Content-Type: application/json", data => $obj, }); return HUEBridge_ProcessResponse($hash,from_json($data)); HUEBridge_dispatch( {hash=>$hash,chash=>$chash,type=>$path},$err,$data ); } else { Log3 $name, 4, "using HttpUtils_NonblockingGet: $method ". ($path?$path:''); my($err,$data) = HttpUtils_NonblockingGet({ url => $url, timeout => 10, method => $method, noshutdown => 1, header => "Content-Type: application/json", data => $obj, hash => $hash, chash => $chash, type => $path, callback => \&HUEBridge_dispatch, }); return undef; } } sub HUEBridge_dispatch($$$;$) { my ($param, $err, $data,$json) = @_; my $hash = $param->{hash}; my $name = $hash->{NAME}; #Log3 $name, 5, "HUEBridge_dispatch"; if( $err ) { Log3 $name, 2, "$name: http request failed: $err"; } elsif( $data || $json ) { if( !$data && !$json ) { Log3 $name, 2, "$name: empty answer received"; return undef; } elsif( $data && $data !~ m/^[\[{].*[\]}]$/ ) { Log3 $name, 2, "$name: invalid json detected: $data"; return undef; } my $queryAfterSet = AttrVal( $name,'queryAfterSet', 0 ); $json = from_json($data) if( !$json ); my $type = $param->{type}; if( ref($json) eq 'ARRAY' ) { HUEBridge_ProcessResponse($hash,$json) if( !$queryAfterSet ); if( defined($json->[0]->{error})) { my $error = $json->[0]->{error}->{'description'}; $hash->{STATE} = $error; Log3 $name, 3, $error; } #return ($json->[0]); } if( $hash == $param->{chash} ) { if( !defined($type) ) { HUEBridge_Parse($hash,$json->{config}); if( defined($json->{sensors}) ) { my $sensors = $json->{sensors}; foreach my $id ( keys %{$sensors} ) { my $code = $name ."-S". $id; my $chash = $modules{HUEDevice}{defptr}{$code}; if( defined($chash) ) { HUEDevice_Parse($chash,$sensors->{$id}); } else { Log3 $name, 4, "$name: message for unknow sensor received: $code"; } } } if( defined($json->{groups}) ) { my $groups = $json->{groups}; foreach my $id ( keys %{$groups} ) { my $code = $name ."-G". $id; my $chash = $modules{HUEDevice}{defptr}{$code}; if( defined($chash) ) { HUEDevice_Parse($chash,$groups->{$id}); } else { Log3 $name, 2, "$name: message for unknow group received: $code"; } } } $type = 'lights'; $json = $json->{lights}; } if( $type eq 'lights' ) { my $lights = $json; foreach my $id ( keys %{$lights} ) { my $code = $name ."-". $id; my $chash = $modules{HUEDevice}{defptr}{$code}; if( defined($chash) ) { HUEDevice_Parse($chash,$lights->{$id}); } else { Log3 $name, 2, "$name: message for unknow device received: $code"; } } } elsif( $type =~ m/^config$/ ) { HUEBridge_Parse($hash,$json); } else { Log3 $name, 2, "$name: message for unknow type received: $type"; Log3 $name, 4, Dumper $json; } } elsif( $type =~ m/^lights\/(\d*)$/ ) { HUEDevice_Parse($param->{chash},$json); } elsif( $type =~ m/^groups\/(\d*)$/ ) { HUEDevice_Parse($param->{chash},$json); } elsif( $type =~ m/^sensors\/(\d*)$/ ) { HUEDevice_Parse($param->{chash},$json); } elsif( $type =~ m/^lights\/(\d*)\/state$/ ) { if( $queryAfterSet ) { my $chash = $param->{chash}; if( $chash->{helper}->{update_timeout} ) { RemoveInternalTimer($chash); InternalTimer(gettimeofday()+1, "HUEDevice_GetUpdate", $chash, 0); } else { RemoveInternalTimer($chash); HUEDevice_GetUpdate( $chash ); } } } elsif( $type =~ m/^groups\/(\d*)\/action$/ ) { my $chash = $param->{chash}; if( $chash->{helper}->{update_timeout} ) { RemoveInternalTimer($chash); InternalTimer(gettimeofday()+1, "HUEDevice_GetUpdate", $chash, 0); } else { RemoveInternalTimer($chash); HUEDevice_GetUpdate( $chash ); } } else { Log3 $name, 2, "$name: message for unknow type received: $type"; Log3 $name, 4, Dumper $json; } } } #adapted version of the CustomGetFileFromURL subroutine from HttpUtils.pm sub HUEBridge_HTTP_Request($$$@) { my ($quiet, $url, $method, $timeout, $data, $noshutdown) = @_; $timeout = 4.0 if(!defined($timeout)); my $displayurl= $quiet ? "" : $url; if($url !~ /^(http|https):\/\/([^:\/]+)(:\d+)?(\/.*)$/) { Log3 undef, 1, "HUEBridge_HTTP_Request $displayurl: malformed or unsupported URL"; return undef; } my ($protocol,$host,$port,$path)= ($1,$2,$3,$4); if(defined($port)) { $port =~ s/^://; } else { $port = ($protocol eq "https" ? 443: 80); } $path= '/' unless defined($path); my $conn; if($protocol eq "https") { eval "use IO::Socket::SSL"; if($@) { Log3 undef, 1, $@; } else { $conn = IO::Socket::SSL->new(PeerAddr=>"$host:$port", Timeout=>$timeout); } } else { $conn = IO::Socket::INET->new(PeerAddr=>"$host:$port", Timeout=>$timeout); } if(!$conn) { Log3 undef, 1, "HUEBridge_HTTP_Request $displayurl: Can't connect to $protocol://$host:$port"; undef $conn; return undef; } $host =~ s/:.*//; #my $hdr = ($data ? "POST" : "GET")." $path HTTP/1.0\r\nHost: $host\r\n"; my $hdr = $method." $path HTTP/1.0\r\nHost: $host\r\n"; if(defined($data)) { $hdr .= "Content-Length: ".length($data)."\r\n"; $hdr .= "Content-Type: application/json"; } $hdr .= "\r\n\r\n"; syswrite $conn, $hdr; syswrite $conn, $data if(defined($data)); shutdown $conn, 1 if(!$noshutdown); my ($buf, $ret) = ("", ""); $conn->timeout($timeout); for(;;) { my ($rout, $rin) = ('', ''); vec($rin, $conn->fileno(), 1) = 1; my $nfound = select($rout=$rin, undef, undef, $timeout); if($nfound <= 0) { Log3 undef, 1, "HUEBridge_HTTP_Request $displayurl: Select timeout/error: $!"; undef $conn; return undef; } my $len = sysread($conn,$buf,65536); last if(!defined($len) || $len <= 0); $ret .= $buf; } $ret=~ s/(.*?)\r\n\r\n//s; # Not greedy: switch off the header. my @header= split("\r\n", $1); my $hostpath= $quiet ? "" : $host . $path; Log3 undef, 5, "HUEBridge_HTTP_Request $displayurl: Got data, length: ".length($ret); if(!length($ret)) { Log3 undef, 4, "HUEBridge_HTTP_Request $displayurl: Zero length data, header follows..."; for (@header) { Log3 undef, 4, "HUEBridge_HTTP_Request $displayurl: $_"; } } undef $conn; if($header[0] =~ /^[^ ]+ ([\d]{3})/ && $1 != 200) { my %result = { error => "error: $1" }; return \%result; } return $ret; } 1; =pod =begin html

HUEBridge

    Module to access the bridge of the phillips hue lighting system.

    The actual hue bulbs, living colors or living whites devices are defined as HUEDevice devices.

    All newly found devices and groups are autocreated at startup and added to the room HUEDevice.

    Notes:
    • This module needs JSON.
      Please install with 'cpan install JSON' or your method of choice.
    • autocreate only works for the first bridge. devices on other bridges have to be manualy defined.


    Define
      define <name> HUEBridge [<host>] [<interval>]

      Defines a HUEBridge device with address <host>.

      If [<host>] is not given the module will try to autodetect the bridge with the hue portal services.

      The bridge status will be updated every <interval> seconds. The default and minimum is 60.

      After a new bridge is created the pair button on the bridge has to be pressed.

      Examples:
        define bridge HUEBridge 10.0.1.1

    Get
    • devices
      list the devices known to the bridge.
    • groups
      list the groups known to the bridge.
    • scenes
      list the scenes known to the bridge.
    • sensors
      list the sensors known to the bridge.
    • whitelist
      list the whitlist of the bridge.

    Set
    • autocreate
      Create fhem devices for all bridge devices.
    • autodetect
      Initiate the detection of new ZigBee devices. After aproximately one minute any newly detected devices can be listed with get devices and the corresponding fhem devices can be created by set autocreate.
    • delete <name>|<id>
      Deletes the given device in the bridge and deletes the associated fhem device.
    • creategroup <name> <lights>
      Create a group out of <lights> in the bridge. The lights are given as a comma sparated list of fhem device names or bridge light numbers.
    • deletegroup <name>|<id>
      Deletes the given group in the bridge and deletes the associated fhem device.
    • savescene <id> <name> <lights>
      Create a scene from the current state of <lights> in the bridge. The lights are given as a comma sparated list of fhem device names or bridge light numbers.
    • scene <id>
      Recalls the scene with the given id.
    • modifyscene <id> <light> <light-args>
      Modifys the given scene in the bridge.
    • deletwhitelist <key>
      Deletes the given key from the whitelist in the bridge.
    • touchlink
      perform touchlink action
    • statusRequest
      Update bridge status.
    • swupdate
      Update bridge firmware. This command is only available if a new firmware is available (indicated by updatestate with a value of 2. The version and release date is shown in the reading swupdate.
      A notify of the form define HUEUpdate notify bridge:swupdate.* {...} can be used to be informed about available firmware updates.


=end html =cut