2
0
mirror of https://github.com/fhem/fhem-mirror.git synced 2025-04-05 17:48:44 +00:00

Initial commit of 00_MAXLAN.pm and 10_MAX.pm

00_MAXLAN.pm: update doc

10_MAX.pm: update doc

00_MAXLAN.pm: Fix syntax error

00_MAXLAN.pm: remove reference to hmId

00_MAXLAN.pm: also dispatch define events from metadata, because shutter does not have configdata

10_MAX.pm: allow setting temp to "on" and "off"

10_MAX.pm: fix doc formatting

00_MAXLAN.pm: Add two set commands: raw and clock

00_MAXLAN.pm: parse current until time

00_MAXLAN.pm: send quit in Undef

00_MAXLAN.pm: beautifying

Display ctrl mode (auto/manual/temporary)

Allow setting a temporary temperature

MAX: change event delimiter to handle time

Fix bug where only one room could be handled

MAX: parse "S:" response

List roomid, send correct roomid with desired-temp

MAX: show roomname for devices

MAX: add troubleshooting when device send temperature 0

00_MAXLAN: Fix parsing of "until" date: time and month were broken

00_MAXLAN: show time offset on start

Show more state info

Set timezone to floating because local cannot be determined on fritzbox

dispatch CubeClockState only if rfaddr is known

Read valve position of HeatingThermostat by gaggi (g.sturm@sturm-busse.de)

MAX: notify if 1% percent rule is oversteped and if queue is full

Do not crash on empty Metadata response

change inital label from "initalized" to "waiting for data"

Auto reconnect if connecting failed

defer setting clock until after complete hello

cleanup

Lower polling interval to 10 seconds

Remove ReadyFn, we already got a FD

really defer setting the clock

valveposition is in percent

rename serialNr to serial

check for matching magic/version, be more robust while parsing metadata

implement SendMetadata

Add missing devices during C: response

Fill SetRoomId

Make ReadAnswer really wait for one

Make valveposition a reading/show possible settings in WebUI (by wollet42, wton1@gmx.de)

Remove DateTime module

rename room to group: fhem already has a room concept

Set temperature individually, not for the whole group

Implement setting of groupid

Remove display of groupname, it does not say anything

generate events for HeatingThermostat's readings

Switched to ReadingsBulkUpdate (thanks to tobias.faust, tobias.faust@gmx.net)

Implement interfaces: this changes the setter desired-temp to desiredTemperature

Correct some Log messages

doc: make clear that MAXLAN uses the cube exclusively

Fix webinterface showing old name for desiredTemperature

add temp4valve4.gplot for Max! HeatingThermostats

Move SetTemperature to backend, allow eco/comfort

Show eco/comfort temperature from device

Support event-on-{update,change}-reading

Correct computation of time difference (thanks Andy Fuchs)

gplot: Corrected axis range

Allow configuring of a lot of device parameters

Add optional pollintervall to define (thanks to wollet42)

Return message up to UI

Minor fixes

Fix setting desiredTemperature

Allow cancel of pairmode

gplot: better scaling (by wollet42) and steps instead of interpolation

Continue even if L: is empty

add factoryReset

request config after new device paired

Updated CUL_MAX

Correct parsing of flags

parse measured(!) temperature

Unify logic for MAXLAN and CUL_MAX into MAX

Show some unused bits

Correctly show until time only in temporary mode

Fix measured temp

Fix checks for existance

Return list of valid values for desiredTemperature

Require a value with desiredTemperature and don't show (manual)

Update doc text

Update CUL_MAX

MAXLAN: dispatch define before configuration event

Update plot to show measured temperature, too

git-svn-id: https://svn.fhem.de/fhem/trunk@2144 2b470e98-0d58-463d-a4d8-8e2adae1ed80
This commit is contained in:
mgehre 2012-11-19 23:03:18 +00:00
parent df60171240
commit bc95538e6b
4 changed files with 1248 additions and 0 deletions

712
fhem/FHEM/00_MAXLAN.pm Executable file
View File

@ -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 <name> 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
<a name="MAXLAN"></a>
<h3>MAXLAN</h3>
<ul>
<tr><td>
The MAXLAN is the fhem module for the eQ-3 MAX! Cube LAN Gateway.
<br><br>
The fhem module makes the MAX! "bus" accessible to fhem, automatically detecting paired MAX! devices. (The devices themselves are handled by the <a href="#MAX">MAX</a> module).<br>
The MAXLAN module keeps a persistant connection to the cube. The cube only allows one connection at a time, so neither the Max! Software or the
Max! internet portal can be used at the same time.
<br>
<a name="MAXLANdefine"></a>
<b>Define</b>
<ul>
<code>define &lt;name&gt; MAXLAN &lt;ip-address&gt;[:port] [&lt;pollintervall&gt;]</code><br>
<br>
port is 62910 by default. (If your Cube listens on port 80, you have to update the firmware with
the official MAX! software).
If the ip-address is called none, then no device will be opened, so you
can experiment without hardware attached.<br>
The optional parameter &lt;pollintervall&gt; defines the time in seconds between each polling of data from the cube.<br>
</ul>
<br>
<a name="MAXLANset"></a>
<b>Set</b>
<ul>
<li>pairmode [&lt;n&gt;,cancel]<br>
Sets the cube into pairing mode for &lt;n&gt; seconds (default is 60s ) where it can be paired with other devices (Thermostats, Buttons, etc.). You also have to set the other device into pairing mode manually. (For Thermostats, this is pressing the "Boost" button for 3 seconds, for example).
Setting pairmode to "cancel" puts the cube out of pairing mode.</li>
<li>raw &lt;data&gt;<br>
Sends the raw &lt;data&gt; to the cube.</li>
<li>clock<br>
Sets the internal clock in the cube to the current system time of fhem's machine. You can add<br>
<code>set ml clock</code><br>
to your fhem.cfg to do this automatically on startup.</li>
<li>factorReset<br>
Reset the cube to factory defaults.</li>
</ul>
<br>
<a name="MAXLANget"></a>
<b>Get</b>
<ul>
N/A
</ul>
<br>
<br>
<a name="MAXLANattr"></a>
<b>Attributes</b>
<ul>
<li><a href="#do_not_notify">do_not_notify</a></li><br>
<li><a href="#attrdummy">dummy</a></li><br>
<li><a href="#loglevel">loglevel</a></li><br>
<li><a href="#addvaltrigger">addvaltrigger</a></li><br>
</ul>
</ul>
=end html
=cut

373
fhem/FHEM/10_MAX.pm Executable file
View File

@ -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 <name> 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
<a name="MAX"></a>
<h3>MAX</h3>
<ul>
Devices from the eQ-3 MAX! group.<br>
When heating thermostats show a temperature of zero degrees, they didn't yet send any data to the cube. You can
force the device to send data to the cube by physically setting a temperature directly at the device (not through fhem).
<br><br>
<a name="MAXdefine"></a>
<b>Define</b>
<ul>
<code>define &lt;name&gt; MAX &lt;type&gt; &lt;addr&gt;</code>
<br><br>
Define an MAX device of type &lt;type&gt; and rf address &lt;addr&gt.
The &lt;type&gt; is one of Cube, HeatingThermostat, HeatingThermostatPlus, WallMountedThermostat, ShutterContact, PushButton.
The &lt;addr&gt; is a 6 digit hex number.
You should never need to specify this by yourself, the <a href="#autocreate">autocreate</a> module will do it for you.<br>
It's advisable to set event-on-change-reading, like
<code>attr Heater_0 event-on-change-reading battery,desiredTemperature,valveposition</code>
because the polling mechanism will otherwise create events every 10 seconds.
Example:
<ul>
<code>define switch1 MAX PushButton ffc545</code><br>
</ul>
</ul>
<br>
<a name="MAXset"></a>
<b>Set</b>
<ul>
<li>desiredTemperature &lt;value&gt; [until &lt;date&gt;]<br>
For devices of type HeatingThermostat only. &lt;value&gt; maybe one of
<ul>
<li>degree celcius between 3.5 and 30.5 in 0.5 degree steps</li>
<li>"on" or "off" correspondig to 30.5 and 4.5 degree celcius</li>
<li>"eco" or "comfort" using the eco/comfort temperature set on the device (just as the right-most physical button on the device itself does)</li>
<li>"auto", where the weekly program saved on the thermostat is processed</li>
</ul>
All values but "auto" maybe accompanied by the "until" clause, with &lt;data&gt; in format "dd.mm.yyyy HH:MM" (minutes may only be "30" or "00"!)
to set a temporary temperature until that date/time. Make sure that the cube has valid system time!</li>
<li>groupid &lt;id&gt;<br>
For devices of type HeatingThermostat only.
Writes the given group id the device's memory. It is usually not necessary to change this.</li>
<li>ecoTemperature &lt;value&gt;<br>
For devices of type HeatingThermostat only. Writes the given eco temperature to the device's memory. It can be activated by pressing the rightmost physical button on the device.</li>
<li>comfortTemperature &lt;value&gt;<br>
For devices of type HeatingThermostat only. Writes the given comfort temperature to the device's memory. It can be activated by pressing the rightmost physical button on the device.</li>
<li>temperatureOffset &lt;value&gt;<br>
For devices of type HeatingThermostat only. Writes the given temperature offset to the device's memory. The thermostat tries to match desiredTemperature to (measuredTemperature+temperatureOffset). Usually, the measured temperature is a bit higher than the overall room temperature (due to closeness to the heater), so one uses a small negative offset. Must be between -3.5 and 3.5 degree.</li>
<li>minimumTemperature &lt;value&gt;<br>
For devices of type HeatingThermostat only. Writes the given minimum temperature to the device's memory. It confines the temperature that can be manually set on the device.</li>
<li>maximumTemperature &lt;value&gt;<br>
For devices of type HeatingThermostat only. Writes the given maximum temperature to the device's memory. It confines the temperature that can be manually set on the device.</li>
<li>windowOpenTemperature &lt;value&gt;<br>
For devices of type HeatingThermostat only. Writes the given window open temperature to the device's memory. That is the temperature the heater will temporarily set if an open window is detected.</li>
<li>windowOpenDuration &lt;value&gt;<br>
For devices of type HeatingThermostat only. Writes the given window open duration to the device's memory. That is the duration the heater will temporarily set the window open temperature if an open window is detected by a rapid temperature decrease. (Not used if open window is detected by ShutterControl. Must be between 0 and 60 minutes in multiples of 5.</li>
</ul>
<br>
<a name="MAXget"></a>
<b>Get</b> <ul>N/A</ul><br>
<a name="MAXattr"></a>
<b>Attributes</b>
<ul>
<li><a href="#event-on-update-reading">event-on-update-reading</a></li>
<li><a href="#event-on-change-reading">event-on-change-reading</a></li>
<li><a href="#eventMap">eventMap</a></li>
<li><a href="#IODev">IODev</a></li>
<li><a href="#loglevel">loglevel</a></li>
<li><a href="#do_not_notify">do_not_notify</a></li>
<li><a href="#ignore">ignore</a></li>
</ul>
<br>
<a name="MAXevents"></a>
<b>Generated events:</b>
<ul>
<li>desiredTemperature<br>Only for HeatingThermostat</li>
<li>valveposition<br>Only for HeatingThermostat</li>
<li>battery</li>
<li>temperature<br>The measured(!) temperature, only for HeatingThermostat</li>
</ul>
</ul>
=end html
=cut

129
fhem/FHEM/14_CUL_MAX.pm Normal file
View File

@ -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 <name> CUL_MAX <srdAddr>" 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
<a name="CUL_MAX"></a>
<h3>CUL_MAX</h3>
<ul>
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 <code>attr CUL0 rfmode MAX</code>.<br>
You should also set a (random) source address for this module by <code>attr CUL0 srcAddrMAX <6-digit-hex></code>
<br><br>
<a name="CUL_MAXdefine"></a>
<b>Define</b>
<ul>N/A</ul>
<br>
<a name="CUL_MAXset"></a>
<b>Set</b> <ul>N/A</ul><br>
<a name="CUL_MAXget"></a>
<b>Get</b> <ul>N/A</ul><br>
<a name="CUL_MAXattr"></a>
<b>Attributes</b>
<ul>
<li><a href="#ignore">ignore</a></li><br>
<li><a href="#do_not_notify">do_not_notify</a></li><br>
<li><a href="#showtime">showtime</a></li><br>
<li><a href="#loglevel">loglevel</a></li><br>
</ul>
<br>
<a name="CUL_MAXevents"></a>
<b>Generated events:</b>
<ul>N/A</ul>
<br>
</ul>
=end html
=cut

View File

@ -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 <SIZE> crop
set terminal png transparent size <SIZE> crop
set output '<OUT>.png'
set xdata time
set timefmt "%Y-%m-%d_%H:%M:%S"
set xlabel " "
set ytics nomirror
set y2tics
#set ytics
set title '<TL>'
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' <IN>"\
using 1:4 axes x1y2 title 'Desired Temperature' with steps,\
"< egrep 'temperature' <IN>"\
using 1:4 axes x1y2 title 'Ist Temperature' with lines,\
"< egrep 'valveposition' <IN>"\
using 1:4 axes x1y1 title 'Valve position (%)' with steps