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:
parent
df60171240
commit
bc95538e6b
712
fhem/FHEM/00_MAXLAN.pm
Executable file
712
fhem/FHEM/00_MAXLAN.pm
Executable 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 <name> MAXLAN <ip-address>[:port] [<pollintervall>]</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 <pollintervall> 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 [<n>,cancel]<br>
|
||||
Sets the cube into pairing mode for <n> 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 <data><br>
|
||||
Sends the raw <data> 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
373
fhem/FHEM/10_MAX.pm
Executable 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 <name> MAX <type> <addr></code>
|
||||
<br><br>
|
||||
|
||||
Define an MAX device of type <type> and rf address <addr>.
|
||||
The <type> is one of Cube, HeatingThermostat, HeatingThermostatPlus, WallMountedThermostat, ShutterContact, PushButton.
|
||||
The <addr> 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 <value> [until <date>]<br>
|
||||
For devices of type HeatingThermostat only. <value> 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 <data> 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 <id><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 <value><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 <value><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 <value><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 <value><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 <value><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 <value><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 <value><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
129
fhem/FHEM/14_CUL_MAX.pm
Normal 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
|
34
fhem/www/gplot/temp4valve4.gplot
Normal file
34
fhem/www/gplot/temp4valve4.gplot
Normal 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
|
Loading…
x
Reference in New Issue
Block a user