diff --git a/fhem/FHEM/00_MAXLAN.pm b/fhem/FHEM/00_MAXLAN.pm new file mode 100755 index 000000000..40bb0e9ca --- /dev/null +++ b/fhem/FHEM/00_MAXLAN.pm @@ -0,0 +1,712 @@ +############################################## +# $Id$ +# Written by Matthias Gehre, M.Gehre@gmx.de, 2012 +package main; + +use strict; +use warnings; +use MIME::Base64; +use Data::Dumper; +use POSIX; + +sub MAXLAN_Parse($$); +sub MAXLAN_ParseDateTime($$$); +sub MAXLAN_Read($); +sub MAXLAN_Write($$); +sub MAXLAN_ReadAnswer($); +sub MAXLAN_SimpleWrite(@); +sub MAXLAN_Poll($); +sub MAXLAN_SendDeviceCmd($$); +sub MAXLAN_RequestConfiguration($$); + +my %device_types = ( + 0 => "Cube", + 1 => "HeatingThermostat", + 2 => "HeatingThermostatPlus", + 3 => "WallMountedThermostat", + 4 => "ShutterContact", + 5 => "PushButton" +); + +my @boost_durations = (0, 5, 10, 15, 20, 25, 30, 60); + +#Time after which we reconnect after a failed connection attempt +my $reconnect_interval = 5; #seconds + +my $metadata_magic = 0x56; +my $metadata_version = 2; + +my $defaultPollInterval = 10; + +sub +MAXLAN_Initialize($) +{ + my ($hash) = @_; + + require "$attr{global}{modpath}/FHEM/DevIo.pm"; + +# Provider + $hash->{ReadFn} = "MAXLAN_Read"; + $hash->{WriteFn} = "MAXLAN_Write"; + $hash->{SetFn} = "MAXLAN_Set"; + $hash->{Clients} = ":MAX:"; + my %mc = ( + "1:MAX" => "^MAX", + ); + $hash->{MatchList} = \%mc; + +# Normal devices + $hash->{DefFn} = "MAXLAN_Define"; + $hash->{UndefFn} = "MAXLAN_Undef"; + $hash->{AttrList}= "do_not_notify:1,0 dummy:1,0 " . + "loglevel:0,1,2,3,4,5,6 addvaltrigger "; +} + +##################################### +sub +MAXLAN_Define($$) +{ + my ($hash, $def) = @_; + my @a = split("[ \t][ \t]*", $def); + + if(@a < 3 or @a > 4) { + my $msg = "wrong syntax: define MAXLAN ip[:port] [pollintervall]"; + Log 2, $msg; + return $msg; + } + DevIo_CloseDev($hash); + + my $name = $a[0]; + my $dev = $a[2]; + $dev .= ":62910" if($dev !~ m/:/ && $dev ne "none" && $dev !~ m/\@/); + + if($dev eq "none") { + Log 1, "$name device is none, commands will be echoed only"; + $attr{$name}{dummy} = 1; + return undef; + } + $hash->{PARTIAL} = ""; + $hash->{DeviceName} = $dev; + $hash->{INTERVAL} = @a > 3 ? $a[3] : $defaultPollInterval; + #This interface is shared with 14_CUL_MAX.pm + $hash->{SendDeviceCmd} = \&MAXLAN_SendDeviceCmd; + + + MAXLAN_Connect($hash); +} + +sub +MAXLAN_Connect($) +{ + my $hash = shift; + + delete($hash->{NEXT_OPEN}); #work around the connection rate limiter in DevIo + my $ret = DevIo_OpenDev($hash, 0, "MAXLAN_DoInit"); + if($hash->{STATE} ne "opened"){ + Log 3, "Scheduling reconnect attempt in $reconnect_interval seconds"; + InternalTimer(gettimeofday()+$reconnect_interval, "MAXLAN_Connect", $hash, 0); + } + return $ret; +} + + +##################################### +sub +MAXLAN_Undef($$) +{ + my ($hash, $arg) = @_; + RemoveInternalTimer($hash); + MAXLAN_Write($hash,"q:"); + DevIo_CloseDev($hash); + return undef; +} + +##################################### +sub +MAXLAN_Set($@) +{ + my ($hash, $device, @a) = @_; + return "\"set MAXLAN\" needs at least one parameter" if(@a < 1); + my ($setting, @args) = @a; + + if($setting eq "pairmode"){ + if(@args > 0 and $args[0] eq "cancel") { + MAXLAN_Write($hash,"x:"); + return MAXLAN_ExpectAnswer($hash,"N:"); + } else { + my $duration = 60; + $duration = $args[0] if(@args > 0); + MAXLAN_Write($hash,"n:".sprintf("%04x",$duration)); + $hash->{STATE} = "pairing"; + } + + }elsif($setting eq "raw"){ + MAXLAN_Write($hash,$args[0]); + + }elsif($setting eq "clock"){ + if(!exists($hash->{rfaddr})){ + Log 5, "Defering the setting of time until after hello"; + $hash->{setTimeOnHello} = 1; + return; + } + + #This encodes the winter/summer timezones, its meaning is not entirely clear + my $timezones = "Q0VUAAAKAAMAAA4QQ0VTVAADAAIAABwg"; + + #The offset was obtained by experiment and is up to 1 minute, I don't know exactly what + #time format the cube uses. Something based on ntp I guess. Maybe this only works in GMT+1? + my $time = time()-946684774; + my $rmsg = "v:".$timezones.",".sprintf("%08x",$time); + MAXLAN_Write($hash,$rmsg); + my $answer = MAXLAN_ReadAnswer($hash); + if($answer ne "A:"){ + Log 1, "Failed to set clock, answer was $answer, expected A:"; + }else{ + Dispatch($hash, "MAX,CubeClockState,$hash->{rfaddr},1", {RAWMSG => $rmsg}); + } + + }elsif($setting eq "factoryReset") { + MAXLAN_RequestReset($hash); + }else{ + return "Unknown argument $setting, choose one of pairmode raw clock factoryReset"; + } + return undef; +} + +sub +MAXLAN_ExpectAnswer($$) +{ + my ($hash,$expectedanswer) = @_; + my $rmsg = MAXLAN_ReadAnswer($hash); + return "Error while receiving" if(!defined($rmsg)); #error is already logged in MAXLAN_ReadAnswer + + my $ret = undef; + if($rmsg !~ m/^$expectedanswer/) { + Log 2, "MAXLAN_ParseAnswer: Got unexpected response, expected $expectedanswer"; + MAXLAN_Parse($hash,$rmsg); + return "Got unexpected response, expected $expectedanswer"; + } + return MAXLAN_Parse($hash,$rmsg); +} + + +##################################### +sub +MAXLAN_ReadAnswer($) +{ + my ($hash) = @_; + + my $data = $hash->{PARTIAL}; + + #Read until we have a complete line + until($data =~ m/\n/) { + my $buf = DevIo_SimpleRead($hash); + if(!defined($buf)){ + Log 1, "MAXLAN_ReadAnswer: error during read"; + return undef; #error occured + } + $data .= $buf; + } + + my $rmsg; + ($rmsg,$data) = split("\n", $data, 2); + $rmsg =~ s/\r//; #remove \r + #save partial data for next read + $hash->{PARTIAL} = $data; + return $rmsg; +} + +my %lhash; + +##################################### +sub +MAXLAN_Write($$) +{ + my ($hash,$msg) = @_; + + MAXLAN_SimpleWrite($hash, $msg); +} + +##################################### +# called from the global loop, when the select for hash->{FD} reports data +sub +MAXLAN_Read($) +{ + my ($hash) = @_; + + my $buf = DevIo_SimpleRead($hash); + return "" if(!defined($buf)); + my $name = $hash->{NAME}; + + my $data = $hash->{PARTIAL}; + $data .= $buf; + + while($data =~ m/\n/) { + my $rmsg; + ($rmsg,$data) = split("\n", $data, 2); + $hash->{PARTIAL} = $data; + $rmsg =~ s/\r//; + MAXLAN_Parse($hash, $rmsg) if($rmsg); + } +} + +sub +MAXLAN_ParseDateTime($$$) +{ + my ($byte1,$byte2,$byte3) = @_; + my $day = $byte1 & 0x1F; + my $month = (($byte1 & 0xE0) >> 4) | ($byte2 >> 7); + my $year = $byte2 & 0x3F; + my $time = ($byte3 & 0x3F); + if($time%2){ + $time = int($time/2).":30"; + }else{ + $time = int($time/2).":00"; + } + return { "day" => $day, "month" => $month, "year" => $year, "time" => $time, "str" => "$day.$month.$year $time" }; +} + +sub +MAXLAN_SendMetadata($) +{ + my $hash = shift; + + if(defined($hash->{metadataVersionMismatch})){ + Log 3,"MAXLAN_SendMetadata: current version of metadata unexpected, not overwriting!"; + return; + } + + my $maxNameLength = 32; + my $maxGroupCount = 20; + my $maxDeviceCount = 140; + + my @groups = @{$hash->{groups}}; + my @devices = @{$hash->{devices}}; + + if(@groups > $maxGroupCount || @devices > $maxDeviceCount) { + Log 1, "MAXLAN_SendMetadata: you got more than $maxGroupCount groups or $maxDeviceCount devices"; + return; + } + + my $metadata = pack("CC",$metadata_magic,$metadata_version); + + $metadata .= pack("C",scalar(@groups)); + foreach(@groups){ + if(length($_->{name}) > $maxNameLength) { + Log 1, "Group name $_->{name} is too long, maximum of $maxNameLength characters allowed"; + return; + } + $metadata .= pack("CC/aH6",$_->{id}, $_->{name}, $_->{masterAddr}); + } + $metadata .= pack("C",scalar(@devices)); + foreach(@devices){ + if(length($_->{name}) > $maxNameLength) { + Log 1, "Device name $_->{name} is too long, maximum of $maxNameLength characters allowed"; + return; + } + $metadata .= pack("CH6a[10]C/aC",$_->{type}, $_->{addr}, $_->{serial}, $_->{name}, $_->{groupid}); + } + + $metadata .= pack("C",1); #dstenables, should always be 1 + my $blocksize = 1900; + + $metadata = encode_base64($metadata,""); + + my $numpackages = ceil(length($metadata)/$blocksize); + for(my $i=0;$i < $numpackages; $i++) { + my $package = substr($metadata,$i*$blocksize,$blocksize); + + MAXLAN_Write($hash,"m:".sprintf("%02d",$i).",".$package); + my $answer = MAXLAN_ReadAnswer($hash); + if($answer ne "A:"){ + Log 1, "SendMetadata got response $answer, expected 'A:'"; + return; + } + } +} + +sub +MAXLAN_Parse($$) +{ + #http://www.domoticaforum.eu/viewtopic.php?f=66&t=6654 + my ($hash, $rmsg) = @_; + + my $name = $hash->{NAME}; + my $ll3 = GetLogLevel($name,3); + my $ll5 = GetLogLevel($name,5); + Log $ll5, "Msg $rmsg"; + my $cmd = substr($rmsg,0,1); # get leading char + my @args = split(',', substr($rmsg,2)); + #Log $ll5, 'args '.join(" ",@args); + + if ($cmd eq 'H'){ #Hello + $hash->{serial} = $args[0]; + $hash->{rfaddr} = $args[1]; + $hash->{fwversion} = $args[2]; + my $dutycycle = 0; + if(@args > 5){ + $dutycycle = $args[5]; + } + my $freememory = 0; + if(@args > 6){ + $freememory = $args[6]; + } + my $cubedatetime = { + year => 2000+hex(substr($args[7],0,2)), + month => hex(substr($args[7],2,2)), + day => hex(substr($args[7],4,2)), + hour => hex(substr($args[8],0,2)), + minute => hex(substr($args[8],2,2)), + }; + my $clockset = hex($args[9]); + #$cubedatetime is only valid if $clockset is 1 + if($clockset) { + my ($sec,$min,$hour,$mday,$mon,$year) = localtime(time); + my $difference = ((((($cubedatetime->{year} - $year-1900)*12 + + $cubedatetime->{month} - $mon-1)*30 + + $cubedatetime->{day} - $mday)*24 + + $cubedatetime->{hour} - $hour)*60 + + $cubedatetime->{minute} - $min); + Log 3, "Cube thinks it is $cubedatetime->{day}.$cubedatetime->{month}.$cubedatetime->{year} $cubedatetime->{hour}:$cubedatetime->{minute}"; + Log 3, "Time difference is $difference minutes"; + } + + Dispatch($hash, "MAX,define,$hash->{rfaddr},Cube,$hash->{serial}", {RAWMSG => $rmsg}); + Dispatch($hash, "MAX,CubeConnectionState,$hash->{rfaddr},1", {RAWMSG => $rmsg}); + Dispatch($hash, "MAX,CubeClockState,$hash->{rfaddr},$clockset", {RAWMSG => $rmsg}); + Log $ll5, "MAXLAN_Parse: Got hello, connection ip $args[4], duty cycle $dutycycle, freememory $freememory, clockset $clockset"; + + } elsif($cmd eq 'M') { + #Metadata, this is basically a readwrite part of the cube's memory. + #I don't think that the cube interprets any of that data. + #One can write to that memory with the "m:" command + #The actual configuration comes with the "C:" response and can be set + #with the "s:" command. + return if(@args < 3); #On virgin devices, we get nothing, not even $magic$version$numgroups$numdevices + + my $bindata = decode_base64($args[2]); + #$version is the version the serialized data format I guess + my ($magic,$version,$numgroups,@groupsdevices); + eval { + ($magic,$version,$numgroups,@groupsdevices) = unpack("CCCXC/(CC/aH6)C/(CH6a[10]C/aC)C",$bindata); + 1; + } or do { + Log 1, "Metadata response is malformed!"; + return; + }; + + if($magic != $metadata_magic || $version != $metadata_version) { + Log 3, "MAXLAN_Parse: magic $magic/version $version are not $metadata_magic/$metadata_version as expected"; + $hash->{metadataVersionMismatch} = 1; + } + + my $daylightsaving = pop(@groupsdevices); #should be always true (=0x01) + + my $i; + $hash->{groups} = (); + for($i=0;$i<3*$numgroups;$i+=3){ + $hash->{groups}[@{$hash->{groups}}]->{id} = $groupsdevices[$i]; + $hash->{groups}[-1]->{name} = $groupsdevices[$i+1]; + $hash->{groups}[-1]->{masterAddr} = $groupsdevices[$i+2]; + } + #After a device is freshly paired, it does not appear in this metadata response, + #we first have to set some metadata for it + $hash->{devices} = (); + for(;$i<@groupsdevices;$i+=5){ + $hash->{devices}[@{$hash->{devices}}]->{type} = $groupsdevices[$i]; + $hash->{devices}[-1]->{addr} = $groupsdevices[$i+1]; + $hash->{devices}[-1]->{serial} = $groupsdevices[$i+2]; + $hash->{devices}[-1]->{name} = $groupsdevices[$i+3]; + $hash->{devices}[-1]->{groupid} = $groupsdevices[$i+4]; + Dispatch($hash, "MAX,define,$hash->{devices}[-1]->{addr},$device_types{$hash->{devices}[-1]->{type}},$hash->{devices}[-1]->{serial},$hash->{devices}[-1]->{groupid}", {RAWMSG => $rmsg}); + } + + Log $ll5, "Got Metadata, hash: ".Dumper($hash); + + }elsif($cmd eq "C"){#Configuration + return if(@args < 2); + my $bindata = decode_base64($args[1]); + my ($len,$addr,$devicetype,$groupid,$firmware,$testresult,$serial) = unpack("CH6CCCCa[10]", $bindata); + + Dispatch($hash, "MAX,define,$addr,$device_types{$devicetype},$serial,$groupid", {RAWMSG => $rmsg}); + #devicetype: Cube = 0, HeatingThermostat = 1, HeatingThermostatPlus = 2, WallMountedThermostat = 3, ShutterContact = 4, PushButton = 5 + #Seems that ShutterContact does not have any configdata + if($devicetype == 0){#Cube + #TODO: there is a lot of data left to interpret + Log $ll5, "len $len, addr $addr, devicetype $devicetype, firmware $firmware, testresult $testresult, groupid $groupid, serial $serial"; + }elsif($devicetype == 1){#HeatingThermostat + my ($comforttemp,$ecotemp,$maxsetpointtemp,$minsetpointtemp,$tempoffset,$windowopentemp,$windowopendur,$boost,$decalcifiction,$maxvalvesetting,$valveoffset) = unpack("CCCCCCCCCCC",substr($bindata,18)); + my $boostValve = ($boost & 0x1F) * 5; + my $boostDuration = $boost_durations[$boost >> 5]; #in minutes + #There is some trailing data missing, which maps to the weekly program + $comforttemp=$comforttemp/2.0; #convert to degree celcius + $ecotemp=$ecotemp/2.0; #convert to degree celcius + $tempoffset = $tempoffset/2.0-3.5; #convert to degree + $maxsetpointtemp=$maxsetpointtemp/2.0; + $minsetpointtemp=$minsetpointtemp/2.0; + $windowopentemp=$windowopentemp/2.0; + $windowopendur=$windowopendur*5; + Log $ll5, "len $len, addr $addr, devicetype $devicetype, groupid $groupid, serial $serial, comfortemp $comforttemp, ecotemp $ecotemp, boostValve $boostValve, boostDuration $boostDuration, tempoffset $tempoffset, $minsetpointtemp minsetpointtemp, maxsetpointtemp $maxsetpointtemp, windowopentemp $windowopentemp, windowopendur $windowopendur"; + Dispatch($hash, "MAX,HeatingThermostatConfig,$addr,$ecotemp,$comforttemp,$boostValve,$boostDuration,$tempoffset,$maxsetpointtemp,$minsetpointtemp,$windowopentemp,$windowopendur", {RAWMSG => $rmsg}); + }elsif($devicetype == 4){#ShutterContact TODO + Log $ll5, "len $len, addr $addr, devicetype $devicetype, firmware $firmware, testresult $testresult, groupid $groupid, serial $serial"; + }else{ #TODO + Log $ll5, "Got configdata for unimplemented devicetype $devicetype: len $len, addr $addr, devicetype $devicetype, groupid $groupid, serial $serial"; + } + + #Check if it is already recorded in devices + my $found = 0; + foreach (@{$hash->{devices}}) { + $found = 1 if($_->{addr} eq $addr); + } + #Add device if it is not already known and not the cube itself + if(!$found && $devicetype != 0){ + $hash->{devices}[@{$hash->{devices}}]->{type} = $devicetype; + $hash->{devices}[-1]->{addr} = $addr; + $hash->{devices}[-1]->{serial} = $serial; + $hash->{devices}[-1]->{name} = "no name"; + $hash->{devices}[-1]->{groupid} = $groupid; + } + + }elsif($cmd eq 'L'){#List + + my $bindata = ""; + $bindata = decode_base64($args[0]) if(@args > 0); + #The L command consists of blocks of states (one for each device) + while(length($bindata)){ + my ($len,$addr,$errframetype,$bits1) = unpack("CH6Ca",$bindata); + my $unkbit1 = vec($bits1,0,1); + my $unkbit2 = vec($bits1,2,1); + my $initialized = vec($bits1,1,1); + my $rferror1 = vec($bits1,3,1); + my $valid = vec($bits1,4,1); + my $unkbit3 = vec($bits1,5,1); + my $unkbit4 = vec($bits1,6,2); + + Log 5, "bindata: ".unpack("H*",substr($bindata,0,$len+1)); #+1 because the len field is not counted + Log 5, "len $len, addr $addr, initialized $initialized, valid $valid, rferror $rferror1, errframetype $errframetype, unkbit ($unkbit1,$unkbit2,$unkbit3,$unkbit4)"; + + my $shash = $modules{MAX}{defptr}{$addr}; + + if(!$shash) { + Log 2, "Got List response for undefined device with addr $addr"; + }elsif($shash->{type} eq "HeatingThermostat"){ + my ($bits2,$valveposition,$temperaturesetpoint,$until1,$until2,$until3) = unpack("aCCCCC",substr($bindata,6)); + my $ctrlmode = vec($bits2, 0, 2); + my $dstsetting = vec($bits2, 3, 1); + my $rferror = vec($bits2, 6, 1); + my $battery = vec($bits2, 7, 1); + + my $untilStr = MAXLAN_ParseDateTime($until1,$until2,$until3)->{str}; + my $curTemp = $until2/10; + #If the control mode is not "temporary", the cube sends the current (measured) temperature + $curTemp = "" if($ctrlmode == 2 || $curTemp == 0); + $untilStr = "" if($ctrlmode != 2); + + $temperaturesetpoint = $temperaturesetpoint/2.0; #convert to degree celcius + Log 5, "battery $battery, rferror $rferror, dstsetting $dstsetting, ctrlmode $ctrlmode, valveposition $valveposition %, temperaturesetpoint $temperaturesetpoint, until $untilStr, curTemp $curTemp"; + + Dispatch($hash, "MAX,HeatingThermostatState,$addr,$temperaturesetpoint,$ctrlmode,$untilStr,$battery,$rferror,$dstsetting,$valveposition,$curTemp", {RAWMSG => $rmsg}); + }elsif($shash->{type} eq "ShutterContact"){ + my $bits2 = substr($bindata,6,1); + my $isopen = vec($bits2,0,2) == 0 ? 0 : 1; + my $unkbit5 = vec($bits2,2,4); + my $rferror = vec($bits2,6,1); + my $battery = vec($bits2,7,1); + Log 5, "ShutterContact isopen $isopen, rferror $rferror, battery $battery, unkbits $unkbit5"; + + Dispatch($hash, "MAX,ShutterContactState,$addr,$isopen,$battery,$rferror", {RAWMSG => $rmsg}); + }else{ + Log 2, "Got status for unimplemented device type $shash->{type}"; + } + $bindata=substr($bindata,$len+1); #+1 because the len field is not counted + } + + if(!$hash->{gothello}) { + # "L:..." is the last response after connection before the cube starts to idle + $hash->{gothello} = 1; + #Handle deferred setting of time (L: is the last response after connection before the cube starts to idle) + if(defined($hash->{setTimeOnHello})) { + MAXLAN_Set($hash,$hash->{NAME},"clock"); + delete $hash->{setTimeOnHello}; + } + #Enable polling timer + InternalTimer(gettimeofday()+$hash->{INTERVAL}, "MAXLAN_Poll", $hash, 0) + } + }elsif($cmd eq "N"){#New device paired + if(@args==0){ + $hash->{STATE} = "initalized"; #pairing ended + return; + } + my ($type, $addr, $serial) = unpack("CH6a[10]", decode_base64($args[0])); + Log 2, "Paired new device, type $device_types{$type}, addr $addr, serial $serial"; + Dispatch($hash, "MAX,define,$addr,$device_types{$type},$serial", {RAWMSG => $rmsg}); + + #After a device has been paired, it automatically appears in the "L" and "C" commands, + MAXLAN_RequestConfiguration($hash,$addr); + } elsif($cmd eq "A"){#Acknowledged + Log 3, "Got stray Acknowledged from cube, this should be read by MAXLAN_ReadAnswer"; + + } elsif($cmd eq "S"){#Response to s: + my $dutycycle = hex($args[0]); #number of command send over the air + my $discarded = $args[1]; + my $freememoryslot = $args[2]; + Log 5, "dutycyle $dutycycle, freememoryslot $freememoryslot"; + + Log 3, "1% rule: we sent too much, cmd is now in queue" if($dutycycle == 100 && $freememoryslot > 0); + Log 3, "1% rule: we sent too much, queue is full, cmd discarded" if($dutycycle == 100 && $freememoryslot == 0); + Log 3, "Command was discarded" if($discarded); + return "Command was discarded" if($discarded); + } else { + Log $ll5, "$name Unknown command $cmd"; + return "Unknown command $cmd"; + } + return undef; +} + + +######################## +sub +MAXLAN_SimpleWrite(@) +{ + my ($hash, $msg) = @_; + my $name = $hash->{NAME}; + + Log GetLogLevel($name,5), 'MAXLAN_SimpleWrite: '.$msg; + + $msg .= "\r\n"; + + my $ret = syswrite($hash->{TCPDev}, $msg); + #TODO: none of those conditions detect if the connection is actually lost! + if(!$hash->{TCPDev} || !defined($ret) || !$hash->{TCPDev}->connected) { + Log GetLogLevel($name,1), 'MAXLAN_SimpleWrite failed'; + DevIo_CloseDev($hash); + RemoveInternalTimer($hash); + MAXLAN_Connect($hash); + } +} + +######################## +sub +MAXLAN_DoInit($) +{ + my ($hash) = @_; + $hash->{gothello} = 0; + return undef; +} + +sub +MAXLAN_RequestList($) +{ + my $hash = shift; + MAXLAN_Write($hash, "l:"); + return MAXLAN_ExpectAnswer($hash, "L:"); +} + +##################################### +sub +MAXLAN_Poll($) +{ + my $hash = shift; + + return if(!$hash->{FD}); + + if(!defined(MAXLAN_RequestList($hash))) { + InternalTimer(gettimeofday()+$hash->{INTERVAL}, "MAXLAN_Poll", $hash, 0); + } else { + Log 1, "MAXLAN_Poll: Did not get any answer"; + } +} + +sub +MAXLAN_RequestConfiguration($$) +{ + my ($hash,$addr) = @_; + MAXLAN_Write($hash,"c:$addr"); + MAXLAN_ExpectAnswer($hash, "C:"); +} + +#Sends command to a device and waits for acknowledgment +sub +MAXLAN_SendDeviceCmd($$) +{ + my ($hash,$payload) = @_; + MAXLAN_Write($hash,"s:".encode_base64($payload,"")); + return MAXLAN_ExpectAnswer($hash, "S:"); +} + +#Resets the cube (what does that acutally do? Factory reset?) +sub +MAXLAN_RequestReset($) +{ + my $hash = shift; + MAXLAN_Write($hash,"a:"); + MAXLAN_ExpectAnswer($hash, "A:"); +} + +1; + +=pod +=begin html + + +

MAXLAN

+ + +=end html +=cut diff --git a/fhem/FHEM/10_MAX.pm b/fhem/FHEM/10_MAX.pm new file mode 100755 index 000000000..1f8ce3aed --- /dev/null +++ b/fhem/FHEM/10_MAX.pm @@ -0,0 +1,373 @@ +############################################## +# $Id$ +# Written by Matthias Gehre, M.Gehre@gmx.de, 2012 +package main; + +use strict; +use warnings; +use MIME::Base64; +use Data::Dumper; + +sub MAX_Define($$); +sub MAX_Initialize($); +sub MAX_Parse($$); +sub MAX_Set($@); +sub MAX_MD15Cmd($$$); +sub MAX_DateTime2Internal($); + +my @ctrl_modes = ( "auto", "manual", "temporary" ); + +my %interfaces = ( + "Cube" => undef, + "HeatingThermostat" => "thermostat;battery;temperature", + "HeatingThermostatPlus" => "thermostat;battery;temperature", + "WallMountedThermostat" => "thermostat;temperature;battery", + "ShutterContact" => "switch_active;battery", + "PushButton" => "switch_passive;battery" + ); + +sub +MAX_Initialize($) +{ + my ($hash) = @_; + + Log 5, "Calling MAX_Initialize"; + $hash->{Match} = "^MAX"; + $hash->{DefFn} = "MAX_Define"; + $hash->{ParseFn} = "MAX_Parse"; + $hash->{SetFn} = "MAX_Set"; + $hash->{AttrList} = "IODev do_not_notify:1,0 ignore:0,1 dummy:0,1 " . + "showtime:1,0 loglevel:0,1,2,3,4,5,6 event-on-update-reading event-on-change-reading"; + return undef; +} + +############################# +sub +MAX_Define($$) +{ + my ($hash, $def) = @_; + my @a = split("[ \t][ \t]*", $def); + my $name = $hash->{NAME}; + return "wrong syntax: define MAX addr" + if(int(@a)!=4 || $a[3] !~ m/^[A-F0-9]{6}$/i); + + my $type = $a[2]; + my $addr = $a[3]; + Log 5, "Max_define $type with addr $addr "; + $hash->{type} = $type; + $hash->{addr} = $addr; + $hash->{STATE} = "waiting for data"; + $modules{MAX}{defptr}{$addr} = $hash; + + $hash->{internals}{interfaces} = $interfaces{$type}; + + AssignIoPort($hash); + return undef; +} + +sub +MAX_DateTime2Internal($) +{ + my($day, $month, $year, $hour, $min) = ($_[0] =~ /^(\d{2}).(\d{2})\.(\d{4}) (\d{2}):(\d{2})$/); + return (($month&0xE) << 20) | ($day << 16) | (($month&1) << 15) | (($year-2000) << 8) | ($hour*2 + int($min/30)); +} + +############################# +sub +MAX_Set($@) +{ + my ($hash, $devname, @a) = @_; + my ($setting, @args) = @a; + + return "Cannot set without IODev" if(!exists($hash->{IODev})); + + if($setting eq "desiredTemperature"){ + return "can only set desiredTemperature for HeatingThermostat" if($hash->{type} ne "HeatingThermostat"); + return "missing a value" if(@args == 0); + + my $temperature; + my $until = undef; + my $ctrlmode = 1; #0=auto, 1=manual; 2=temporary + + if($args[0] eq "auto") { + #This enables the automatic/schedule mode where the thermostat follows the weekly program + $temperature = 0; + $ctrlmode = 0; #auto + #TODO: auto mode with temperature is also possible + } elsif($args[0] eq "eco") { + $temperature = $hash->{ecoTemperature}; + } elsif($args[0] eq "comfort") { + $temperature = $hash->{comfortTemperature}; + } elsif($args[0] eq "on") { + $temperature = 30.5; + } elsif($args[0] eq "off") { + $temperature = 4.5; + }else{ + $temperature = $args[0]; + } + + if(@args > 1 and $args[1] eq "until" and $ctrlmode == 1) { + $ctrlmode = 2; #temporary + $until = sprintf("%06x",MAX_DateTime2Internal($args[2]." ".$args[3])); + } + + my $groupid = $hash->{groupid}; + $groupid = 0; #comment this line to control the whole group, no only one device + + $temperature = int($temperature*2.0) | ($ctrlmode << 6); #convert to internal representation + my $payload; + if(defined($until)) { + $payload = pack("CCCCCCH6CCH6",0x00,$groupid?0x04:0,0x40,0x00,0x00,0x00,$hash->{addr},$groupid,$temperature,$until); + }else{ + $payload = pack("CCCCCCH6CC" ,0x00,$groupid?0x04:0,0x40,0x00,0x00,0x00,$hash->{addr},$groupid,$temperature); + } + return ($hash->{IODev}{SendDeviceCmd})->($hash->{IODev},$payload); + + }elsif($setting eq "groupid"){ + return "argument needed" if(@args == 0); + + return ($hash->{IODev}{SendDeviceCmd})->($hash->{IODev},pack("CCCCCCH6CC",0x00,0x00,34,0x00,0x00,0x00,$hash->{addr},0x00,$args[0])); + + }elsif( $setting ~~ ["ecoTemperature", "comfortTemperature", "temperatureOffset", "maximumTemperature", "minimumTemperature", "windowOpenTemperature", "windowOpenDuration" ]) { + + return "can only set configuration for HeatingThermostat" if($hash->{type} ne "HeatingThermostat"); + return "Invalid comfortTemperature" if(!exists($hash->{comfortTemperature}) or $hash->{comfortTemperature} < 4.5 or $hash->{comfortTemperature} > 30.5); + return "Invalid ecoTemperature" if(!exists($hash->{ecoTemperature}) or $hash->{ecoTemperature} < 4.5 or $hash->{ecoTemperature} > 30.5); + return "Invalid maximumTemperature" if(!exists($hash->{maximumTemperature}) or $hash->{maximumTemperature} < 4.5 or $hash->{maximumTemperature} > 30.5); + return "Invalid minimumTemperature" if(!exists($hash->{minimumTemperature}) or $hash->{minimumTemperature} < 4.5 or $hash->{minimumTemperature} > 30.5); + return "Invalid windowOpenTemperature" if(!exists($hash->{windowOpenTemperature}) or $hash->{windowOpenTemperature} < 4.5 or $hash->{windowOpenTemperature} > 30.5); + return "Invalid temperatureOffset" if(!exists($hash->{temperatureOffset}) or $hash->{temperatureOffset} < -3.5 or $hash->{temperatureOffset} > 3.5); + return "Invalid windowOpenDuration" if(!exists($hash->{windowOpenDuration}) or $hash->{windowOpenDuration} < 0 or $hash->{windowOpenDuration} > 60); + + $hash->{$setting} = $args[0]; + + my $comfort = int($hash->{comfortTemperature}*2); + my $eco = int($hash->{ecoTemperature}*2); + my $max = int($hash->{maximumTemperature}*2); + my $min = int($hash->{minimumTemperature}*2); + my $offset = int(($hash->{temperatureOffset} + 3.5)*2); + my $windowOpenTemp = int($hash->{windowOpenTemperature}*2); + my $windowOpenTime = int($hash->{windowOpenDuration}/5); + + my $payload = pack("CCCCCCH6C"."CCCCCCC",0x00,0x00,17,0x00,0x00,0x00,$hash->{addr},0x00, + $comfort,$eco,$max,$min,$offset,$windowOpenTemp,$windowOpenTime); + return ($hash->{IODev}{SendDeviceCmd})->($hash->{IODev},$payload); + }else{ + if($hash->{type} eq "HeatingThermostat") { + #Create numbers from 4.5 to 30.5 + my $templist = join(",",map { $_/2 } (9..61)); + return "Unknown argument $setting, choose one of desiredTemperature:eco,comfort,$templist ecoTemperature comfortTemperature temperatureOffset maximumTemperature minimumTemperature windowOpenTemperature windowOpenDuration groupid"; + } else { + return "Unknown argument $setting"; + } + } +} + +############################# +sub +MAX_Parse($$) +{ + my ($hash, $msg) = @_; + my ($MAX,$msgtype,$addr,@args) = split(",",$msg); + + #Find the device with the given addr + my $shash = $modules{MAX}{defptr}{$addr}; + + if(!$shash) + { + if($msgtype eq "define"){ + my $devicetype = $args[0]; + return "UNDEFINED MAX_$addr MAX $devicetype $addr"; + }else{ + return; + } + } + + if($msgtype eq "define"){ + my $devicetype = $args[0]; + Log 1, "Device changed type from $shash->{type} to $devicetype" if($shash->{type} ne $devicetype); + if(@args > 1){ + my $serial = $args[1]; + Log 1, "Device changed serial from $shash->{serial} to $serial" if($shash->{serial} and ($shash->{serial} ne $serial)); + $shash->{serial} = $serial; + } + if(@args > 2){ + my $groupid = $args[2]; + $shash->{groupid} = $groupid; + } + + } elsif($msgtype eq "HeatingThermostatState") { + my $settemp = $args[0]; + my $mode = $ctrl_modes[$args[1]]; + my $until = $args[2]; + my $batterylow = $args[3]; + my $rferror = $args[4]; + my $dstsetting = $args[5]; + my $valveposition = $args[6]; + my $measuredTemperature = ""; + $measuredTemperature = $args[7] if(@args > 7); + + $shash->{mode} = $mode; + $shash->{rferror} = $rferror; + $shash->{dstsetting} = $dstsetting; + if($mode eq "temporary"){ + $shash->{until} = "$until"; + }else{ + delete($shash->{until}); + } + + readingsBeginUpdate($shash); + readingsBulkUpdate($shash, "battery", $batterylow ? "low" : "ok"); + readingsBulkUpdate($shash, "desiredTemperature", $settemp); + readingsBulkUpdate($shash, "valveposition", $valveposition); + if($measuredTemperature ne "") { + readingsBulkUpdate($shash, "temperature", $measuredTemperature); + } + readingsEndUpdate($shash, 0); + + }elsif($msgtype eq "ShutterContactState"){ + my $isopen = $args[0]; + my $batterylow = $args[1]; + my $rferror = $args[2]; + + $shash->{rferror} = $rferror; + + readingsBeginUpdate($shash); + readingsBulkUpdate($shash, "battery", $batterylow ? "low" : "ok"); + readingsBulkUpdate($shash,"onoff",$isopen); + readingsEndUpdate($shash, 0); + + }elsif($msgtype eq "CubeClockState"){ + my $clockset = $args[0]; + $shash->{clocknotset} = !$clockset; + + }elsif($msgtype eq "CubeConnectionState"){ + my $connected = $args[0]; + + readingsSingleUpdate($shash,"connection",$connected,0); + } elsif($msgtype eq "HeatingThermostatConfig") { + + $shash->{ecoTemperature} = $args[0]; + $shash->{comfortTemperature} = $args[1]; + $shash->{boostValveposition} = $args[2]; + $shash->{boostDuration} = $args[3]; + $shash->{temperatureOffset} = $args[4]; + $shash->{maximumTemperature} = $args[5]; + $shash->{minimumTemperature} = $args[6]; + $shash->{windowOpenTemperature} = $args[7]; + $shash->{windowOpenDuration} = $args[8]; + } + + #Build $shash->{STATE} + $shash->{STATE} = "waiting for data"; + if(exists($shash->{READINGS})) { + $shash->{STATE} = $shash->{READINGS}{connection}{VAL} ? "connected" : "not connected" if(exists($shash->{READINGS}{connection})); + $shash->{STATE} = "$shash->{READINGS}{desiredTemperature}{VAL} °C" if(exists($shash->{READINGS}{desiredTemperature})); + $shash->{STATE} = $shash->{READINGS}{onoff}{VAL} ? "opened" : "closed" if(exists($shash->{READINGS}{onoff})); + } + + $shash->{STATE} .= " (clock not set)" if($shash->{clocknotset}); + $shash->{STATE} .= " (auto)" if(exists($shash->{mode}) and $shash->{mode} eq "auto"); + #Don't print this: it's the standard mode + #$shash->{STATE} .= " (manual)" if(exists($shash->{mode}) and $shash->{mode} eq "manual"); + $shash->{STATE} .= " (until ".$shash->{until}.")" if(exists($shash->{mode}) and $shash->{mode} eq "temporary" ); + $shash->{STATE} .= " (battery low)" if($shash->{batterylow}); + $shash->{STATE} .= " (rf error)" if($shash->{rferror}); + + return $shash->{NAME} +} + +1; + +=pod +=begin html + + +

MAX

+ + +=end html +=cut diff --git a/fhem/FHEM/14_CUL_MAX.pm b/fhem/FHEM/14_CUL_MAX.pm new file mode 100644 index 000000000..322ee22e5 --- /dev/null +++ b/fhem/FHEM/14_CUL_MAX.pm @@ -0,0 +1,129 @@ +############################################## +#Written by Matthias Gehre, M.Gehre@gmx.de, 2012 +package main; + +use strict; +use warnings; +use Data::Dumper; + +sub CUL_MAX_SendDeviceCmd($$); + +sub +CUL_MAX_Initialize($) +{ + my ($hash) = @_; + + $hash->{Match} = "^Z"; + $hash->{DefFn} = "CUL_MAX_Define"; + $hash->{UndefFn} = "CUL_MAX_Undef"; + $hash->{ParseFn} = "CUL_MAX_Parse"; + $hash->{AttrList} = "IODev do_not_notify:1,0 ignore:0,1 " . + "showtime:1,0 loglevel:0,1,2,3,4,5,6"; +} + +############################# +sub +CUL_MAX_Define($$) +{ + my ($hash, $def) = @_; + my @a = split("[ \t][ \t]*", $def); + + return "wrong syntax: define CUL_MAX " if(@a<2); + + $hash->{addr} = $a[1]; + $hash->{STATE} = "Defined"; + $hash->{cnt} = 0; + Log 4, "CUL_MAX defined"; + AssignIoPort($hash); + + #This interface is shared with 00_MAXLAN.pm + $hash->{SendDeviceCmd} = \&MAXLAN_SendDeviceCmd; + + return undef; +} + +##################################### +sub +CUL_MAX_Undef($$) +{ + my ($hash, $name) = @_; + return undef; +} + +################################### +my @culHmCmdFlags = ("WAKEUP", "WAKEMEUP", "BCAST", "Bit3", + "BURST", "BIDI", "RPTED", "RPTEN"); +sub +CUL_MAX_Parse($$) +{ + my ($hash, $msg) = @_; + $msg =~ m/Z(..)(..)(..)(..)(......)(......)(.*)/; + my ($len,$msgcnt,$msgFlag,$msgType,$src,$dst,$p) = ($1,$2,$3,$4,$5,$6,$7); + Log 1, "CUL_MAX_Parse: len mismatch" . (length($msg)/2-1) . " != ". hex($len) if(hex($len) != length($msg)/2-1); + my $msgFlLong = ""; + for(my $i = 0; $i < @culHmCmdFlags; $i++) { + $msgFlLong .= ",$culHmCmdFlags[$i]" if(hex($msgFlag) & (1<<$i)); + } + Log 5, "CUL_MAX_Parse: len $len, msgcnt $msgcnt, msgflag $msgFlLong, msgType $msgType, src $src, dst $dst, payload $p"; + return undef; +} + +sub CUL_MAX_SendDeviceCmd($$) +{ + my ($hash,$payload) = @_; + + my $srcAddr = AttrVal($hash->{NAME},"srcAddrMAX","123456"); + + $hash->{cnt} += 1; + substr($payload,3,3) = pack("H6",$srcAddr); + substr($payload,1,0) = pack("C",$hash->{cnt}); + $payload = pack("C",length($payload)) . $payload; + + Log 5, "CUL_MAX_SendDeviceCmd: ". unpack("H*",$payload); +} + +1; + + +=pod +=begin html + + +

CUL_MAX

+
    + The CUL_MAX module interprets MAX! messages received by the CUL. It will be automatically created by autocreate, just make sure + that you set the right rfmode like attr CUL0 rfmode MAX.
    + You should also set a (random) source address for this module by attr CUL0 srcAddrMAX <6-digit-hex> +

    + + + Define +
      N/A
    +
    + + + Set
      N/A

    + + + Get
      N/A

    + + + Attributes + +
    + + + Generated events: +
      N/A
    +
    + +
+ + +=end html +=cut diff --git a/fhem/www/gplot/temp4valve4.gplot b/fhem/www/gplot/temp4valve4.gplot new file mode 100644 index 000000000..8a7c8269f --- /dev/null +++ b/fhem/www/gplot/temp4valve4.gplot @@ -0,0 +1,34 @@ +# Attribute 'small' is useful for gnuplot/-scroll only, +# if plotsize is less than 800,400 +#set terminal png transparent small size crop + +set terminal png transparent size crop + +set output '.png' +set xdata time +set timefmt "%Y-%m-%d_%H:%M:%S" +set xlabel " " +set ytics nomirror +set y2tics +#set ytics +set title '' +set grid xtics y2tics + +set y2label "Temperature in C" +set ylabel "Vavle Position (%)" + + +set y2range [5:25] +set yrange [0:80] + +#FileLog 4:desiredTemperature:4.5: +#FileLog 4:temperature:4.5: +#FileLog 4:valveposition:0: + +plot \ + "< egrep 'desiredTemperature' "\ + using 1:4 axes x1y2 title 'Desired Temperature' with steps,\ + "< egrep 'temperature' "\ + using 1:4 axes x1y2 title 'Ist Temperature' with lines,\ + "< egrep 'valveposition' "\ + using 1:4 axes x1y1 title 'Valve position (%)' with steps