mirror of
https://github.com/fhem/fhem-mirror.git
synced 2025-05-04 22:19:38 +00:00

FBAHA/FBDECT git-svn-id: https://svn.fhem.de/fhem/trunk@2835 2b470e98-0d58-463d-a4d8-8e2adae1ed80
447 lines
10 KiB
Perl
Executable File
447 lines
10 KiB
Perl
Executable File
##############################################
|
|
# $Id: 00_FBAHA.pm 2777 2013-02-20 08:02:01Z rudolfkoenig $
|
|
package main;
|
|
|
|
use strict;
|
|
use warnings;
|
|
use Time::HiRes qw(gettimeofday);
|
|
|
|
sub FBAHA_Read($@);
|
|
sub FBAHA_Write($$$);
|
|
sub FBAHA_ReadAnswer($$$);
|
|
sub FBAHA_Ready($);
|
|
|
|
sub FBAHA_getDevList($$);
|
|
|
|
|
|
sub
|
|
FBAHA_Initialize($)
|
|
{
|
|
my ($hash) = @_;
|
|
|
|
require "$attr{global}{modpath}/FHEM/DevIo.pm";
|
|
|
|
# Provider
|
|
$hash->{ReadFn} = "FBAHA_Read";
|
|
$hash->{WriteFn} = "FBAHA_Write";
|
|
$hash->{ReadyFn} = "FBAHA_Ready";
|
|
$hash->{UndefFn} = "FBAHA_Undef";
|
|
$hash->{ShutdownFn} = "FBAHA_Undef";
|
|
$hash->{ReadAnswerFn} = "FBAHA_ReadAnswer";
|
|
|
|
# Normal devices
|
|
$hash->{DefFn} = "FBAHA_Define";
|
|
$hash->{GetFn} = "FBAHA_Get";
|
|
$hash->{SetFn} = "FBAHA_Set";
|
|
$hash->{AttrList}= "dummy:1,0 loglevel:0,1,2,3,4,5,6 ";
|
|
}
|
|
|
|
my %gets = ("devList"=>1);
|
|
my %sets = ("createDevs"=>1);
|
|
|
|
#####################################
|
|
sub
|
|
FBAHA_Define($$)
|
|
{
|
|
my ($hash, $def) = @_;
|
|
my @a = split("[ \t][ \t]*", $def);
|
|
|
|
if(@a != 3) {
|
|
return "wrong syntax: define <name> FBAHA hostname:2002";
|
|
}
|
|
|
|
DevIo_CloseDev($hash);
|
|
|
|
my $name = $a[0];
|
|
my $dev = $a[2];
|
|
|
|
$hash->{Clients} = ":FBDECT:";
|
|
my %matchList = ( "1:FBDECT" => ".*" );
|
|
$hash->{MatchList} = \%matchList;
|
|
|
|
$hash->{DeviceName} = $dev;
|
|
my $ret = DevIo_OpenDev($hash, 0, "FBAHA_DoInit");
|
|
return $ret;
|
|
}
|
|
|
|
|
|
#####################################
|
|
sub
|
|
FBAHA_Set($@)
|
|
{
|
|
my ($hash, @a) = @_;
|
|
my $name = shift @a;
|
|
|
|
return "set $name needs at least one parameter" if(@a < 1);
|
|
my $type = shift @a;
|
|
|
|
return "Unknown argument $type, choose one of " . join(" ", sort keys %sets)
|
|
if(!defined($sets{$type}));
|
|
|
|
if($type eq "createDevs") {
|
|
my @arg = FBAHA_getDevList($hash,0);
|
|
foreach my $arg (@arg) {
|
|
if($arg =~ m/ID:(\d+).*PROP:(.*)/) {
|
|
my ($i,$p) = ($1,$2,$3);
|
|
Log 1, "UNDEFINED FBDECT_$i FBDECT $i $p";
|
|
}
|
|
}
|
|
}
|
|
|
|
return undef;
|
|
}
|
|
|
|
#####################################
|
|
sub
|
|
FBAHA_Get($@)
|
|
{
|
|
my ($hash, @a) = @_;
|
|
my $name = shift @a;
|
|
|
|
return "get $name needs at least one parameter" if(@a < 1);
|
|
my $type = shift @a;
|
|
|
|
return "Unknown argument $type, choose one of ". join(" ", sort keys %gets)
|
|
if(!defined($gets{$type}));
|
|
|
|
if($type eq "devList") {
|
|
return join("\n", FBAHA_getDevList($hash,0));
|
|
}
|
|
|
|
return undef;
|
|
}
|
|
|
|
sub
|
|
FBAHA_getDevList($$)
|
|
{
|
|
my ($hash, $onlyId) = @_;
|
|
|
|
FBAHA_Write($hash, "05", "00000000"); # CONFIG_REQ
|
|
my ($err, $data) = FBAHA_ReadAnswer($hash, "CONFIG_RSP", "^06");
|
|
return ($err) if($err);
|
|
|
|
$data = substr($data, 32); # Header
|
|
return FBAHA_configInd($data, $onlyId);
|
|
}
|
|
|
|
sub
|
|
FBAHA_configInd($$)
|
|
{
|
|
my ($data, $onlyId) = @_;
|
|
|
|
|
|
my @answer;
|
|
while(length($data) >= 288) {
|
|
my $id = hex(substr($data, 0, 4));
|
|
my $act = hex(substr($data, 4, 2));
|
|
my $typ = hex(substr($data, 8, 4));
|
|
my $lsn = hex(substr($data, 16, 8));
|
|
my $nam = pack("H*",substr($data,24,160)); $nam =~ s/\x0//g;
|
|
|
|
$act = ($act == 2 ? "active" : ($act == 1 ? "inactive" : "removed"));
|
|
|
|
my %tl = ( 2=>"AVM FRITZ!Dect Powerline 546E", 9=>"AVM FRITZ!Dect 200");
|
|
$typ = $tl{$typ} ? $tl{$typ} : "unknown($typ)";
|
|
|
|
my %ll = (7=>"powerMeter",9=>"switch");
|
|
$lsn = join ",", map { $ll{$_} if((1 << $_) & $lsn) } sort keys %ll;
|
|
|
|
my $dlen = hex(substr($data, 280, 8))*2; # DATA MSG
|
|
|
|
push @answer, "NAME:$nam, ID:$id, $act, TYPE:$typ PROP:$lsn"
|
|
if(!$onlyId || $onlyId == $id);
|
|
|
|
if($onlyId && $onlyId == $id) {
|
|
my $mnf = hex(substr($data,184, 4)); # empty/0
|
|
my $idf = substr($data,192,40); # empty/0
|
|
my $frm = substr($data,232,40); # empty/0
|
|
push @answer, " MANUF:$mnf";
|
|
push @answer, " UniqueID:$idf";
|
|
push @answer, " Firmware:$frm";
|
|
push @answer, substr($data, 288, $dlen);
|
|
return @answer;
|
|
}
|
|
$data = substr($data, 288+$dlen); # rest
|
|
}
|
|
return @answer;
|
|
}
|
|
|
|
#####################################
|
|
sub
|
|
FBAHA_DoInit($)
|
|
{
|
|
my $hash = shift;
|
|
my $name = $hash->{NAME};
|
|
Log 1, "FBAHA_DoInit called";
|
|
|
|
FBAHA_Write($hash, "00", "00010001");
|
|
my ($err, $data) = FBAHA_ReadAnswer($hash, "REGISTER", "^01");
|
|
if($err) {
|
|
Log 1, $err;
|
|
$hash->{STATE} = "???";
|
|
return 1;
|
|
}
|
|
|
|
if($data =~ m/^01030010(........)/) {
|
|
$hash->{STATE} = "Initialized";
|
|
$hash->{HANDLE} = $1;
|
|
|
|
} else {
|
|
Log 1, "Got bogus answer for REGISTER request: $data";
|
|
$hash->{STATE} = "???";
|
|
|
|
}
|
|
|
|
FBAHA_Write($hash, "03", "0000028200000000"); # LISTEN
|
|
return undef;
|
|
}
|
|
|
|
#####################################
|
|
sub
|
|
FBAHA_Undef($@)
|
|
{
|
|
my ($hash, $arg) = @_;
|
|
FBAHA_Write($hash, "02", ""); # RELEASE
|
|
DevIo_CloseDev($hash);
|
|
return undef;
|
|
}
|
|
|
|
#####################################
|
|
sub
|
|
FBAHA_Write($$$)
|
|
{
|
|
my ($hash,$fn,$msg) = @_;
|
|
|
|
$msg = sprintf("%s03%04x%s%s", $fn, length($msg)/2+8,
|
|
$hash->{HANDLE} ? $hash->{HANDLE} : "00000000", $msg);
|
|
DevIo_SimpleWrite($hash, $msg, 1);
|
|
}
|
|
|
|
#####################################
|
|
# called from the global loop, when the select for hash->{FD} reports data
|
|
sub
|
|
FBAHA_Read($@)
|
|
{
|
|
my ($hash, $local, $regexp) = @_;
|
|
|
|
my $buf = ($local ? $local : DevIo_SimpleRead($hash));
|
|
return "" if(!defined($buf));
|
|
|
|
my $name = $hash->{NAME};
|
|
my $ll5 = GetLogLevel($name,5);
|
|
|
|
$buf = unpack('H*', $buf);
|
|
my $data = ($hash->{PARTIAL} ? $hash->{PARTIAL} : "");
|
|
|
|
# drop old data
|
|
if($data) {
|
|
$data = "" if(gettimeofday() - $hash->{READ_TS} > 1);
|
|
delete($hash->{READ_TS});
|
|
}
|
|
|
|
Log $ll5, "FBAHA/RAW: $data/$buf";
|
|
$data .= $buf;
|
|
|
|
my $msg;
|
|
while(length($data) >= 16) {
|
|
my $len = hex(substr($data, 4,4))*2;
|
|
last if($len > length($data));
|
|
$msg = substr($data, 0, $len);
|
|
$data = substr($data, $len);
|
|
last if(defined($local) && (!defined($regexp) || ($msg =~ m/$regexp/)));
|
|
|
|
$hash->{"${name}_MSGCNT"}++;
|
|
$hash->{"${name}_TIME"} = TimeNow();
|
|
$hash->{RAWMSG} = $msg;
|
|
my %addvals = (RAWMSG => $msg);
|
|
Dispatch($hash, $msg, \%addvals) if($init_done);
|
|
$msg = undef;
|
|
}
|
|
|
|
$hash->{PARTIAL} = $data;
|
|
$hash->{READ_TS} = gettimeofday() if($data);
|
|
return $msg if(defined($local));
|
|
return undef;
|
|
}
|
|
|
|
#####################################
|
|
# This is a direct read for commands like get
|
|
sub
|
|
FBAHA_ReadAnswer($$$)
|
|
{
|
|
my ($hash, $arg, $regexp) = @_;
|
|
return ("No FD (dummy device?)", undef)
|
|
if(!$hash || ($^O !~ /Win/ && !defined($hash->{FD})));
|
|
my $to = ($hash->{RA_Timeout} ? $hash->{RA_Timeout} : 3);
|
|
|
|
for(;;) {
|
|
return ("Device lost when reading answer for get $arg", undef)
|
|
if(!$hash->{FD});
|
|
my $rin = '';
|
|
vec($rin, $hash->{FD}, 1) = 1;
|
|
my $nfound = select($rin, undef, undef, $to);
|
|
if($nfound < 0) {
|
|
next if ($! == EAGAIN() || $! == EINTR() || $! == 0);
|
|
my $err = $!;
|
|
DevIo_Disconnected($hash);
|
|
return("FBAHA_ReadAnswer $arg: $err", undef);
|
|
}
|
|
return ("Timeout reading answer for get $arg", undef)
|
|
if($nfound == 0);
|
|
my $buf = DevIo_SimpleRead($hash);
|
|
return ("No data", undef) if(!defined($buf));
|
|
|
|
my $ret = FBAHA_Read($hash, $buf, $regexp);
|
|
return (undef, $ret) if(defined($ret));
|
|
}
|
|
}
|
|
|
|
#####################################
|
|
sub
|
|
FBAHA_Ready($)
|
|
{
|
|
my ($hash) = @_;
|
|
|
|
return DevIo_OpenDev($hash, 1, "FBAHA_DoInit")
|
|
if($hash->{STATE} eq "disconnected");
|
|
return 0;
|
|
}
|
|
|
|
1;
|
|
|
|
=pod
|
|
=begin html
|
|
|
|
<a name="FBAHA"></a>
|
|
<h3>FBAHA</h3>
|
|
<ul>
|
|
<b>NOTE:</b> As of FRITZ!OS 5.50 this module is known to work on the FB7390,
|
|
and is known to <b>crash the FB7270</b>.<br><br>
|
|
|
|
This module connects to the AHA server (AVM Home Automation) on a FRITZ!Box.
|
|
It serves as the "physical" counterpart to the <a href="#FBDECT">FBDECT</a>
|
|
devices.
|
|
<br><br>
|
|
<a name="FBAHAdefine"></a>
|
|
<b>Define</b>
|
|
<ul>
|
|
<code>define <name> FBAHA <host>:<port></code>
|
|
<br>
|
|
<br>
|
|
<host> is normally the address of the FRITZ!Box running the AHA server
|
|
(fritz.box or localhost), and :<port> is 2002.
|
|
<br>
|
|
Example:
|
|
<ul>
|
|
<code>define fb1 FBAHA fritz.box:2002</code><br>
|
|
</ul>
|
|
</ul>
|
|
<br>
|
|
|
|
<a name="FBAHAset"></a>
|
|
<b>Set</b>
|
|
<ul>
|
|
<li>createDevs<br>
|
|
create a FHEM device for each DECT device found on the AHA-Host, see also
|
|
get devList.
|
|
</li>
|
|
</ul>
|
|
<br>
|
|
|
|
<a name="FBAHAget"></a>
|
|
<b>Get</b>
|
|
<ul>
|
|
<li>devList<br>
|
|
return a list of devices with short info.
|
|
</li>
|
|
</ul>
|
|
<br>
|
|
|
|
<a name="FBAHAattr"></a>
|
|
<b>Attributes</b>
|
|
<ul>
|
|
<li><a href="#dummy">dummy</a></li>
|
|
<li><a href="#loglevel">loglevel</a></li>
|
|
</ul>
|
|
<br>
|
|
|
|
<a name="FBAHAevents"></a>
|
|
<b>Generated events:</b>
|
|
<ul>
|
|
<li>UNDEFINED FBDECT_$ahaName_${NR} FBDECT $id"
|
|
</li>
|
|
</ul>
|
|
|
|
</ul>
|
|
|
|
|
|
=end html
|
|
|
|
=begin html_DE
|
|
|
|
<a name="FBAHA"></a>
|
|
<h3>FBAHA</h3>
|
|
<ul>
|
|
<b>Achtung:</b> Es ist bekannt, daß dieses Modul mit FRITZ!OS 5.50 auf
|
|
einem FB7390 funktioniert, aber <b>den FB7270 zum Absurz bringt</b>.<br><br>
|
|
|
|
Dieses Modul verbindet sich mit dem AHA (AVM Home Automation) Server auf
|
|
einem FRITZ!Box. Es dient als "physikalisches" Gegenstück zum <a
|
|
href="#FBDECT">FBDECT</a> Modul.
|
|
<br><br>
|
|
<a name="FBAHAdefine"></a>
|
|
<b>Define</b>
|
|
<ul>
|
|
<code>define <name> FBAHA <host>:<port></code>
|
|
<br>
|
|
<br>
|
|
<host> ist normalerweise die Adresse der FRITZ!Box, wo das AHA Server
|
|
läuft (fritz.box oder localhost), <port> ist 2002.
|
|
<br>
|
|
Beispiel:
|
|
<ul>
|
|
<code>define fb1 FBAHA fritz.box:2002</code><br>
|
|
</ul>
|
|
</ul>
|
|
<br>
|
|
|
|
<a name="FBAHAset"></a>
|
|
<b>Set</b>
|
|
<ul>
|
|
<li>createDevs<br>
|
|
legt FHEM Geräte an für jedes auf dem AHA-Server gefundenen DECT
|
|
Eintrag, siehe auch "get devList".
|
|
</li>
|
|
</ul>
|
|
<br>
|
|
|
|
<a name="FBAHAget"></a>
|
|
<b>Get</b>
|
|
<ul>
|
|
<li>devList<br>
|
|
liefert die Liste aller DECT-Einträge der AHA Server zurück, mit
|
|
einem kurzen Info.
|
|
</li>
|
|
</ul>
|
|
<br>
|
|
|
|
<a name="FBAHAattr"></a>
|
|
<b>Attributes</b>
|
|
<ul>
|
|
<li><a href="#dummy">dummy</a></li>
|
|
<li><a href="#loglevel">loglevel</a></li>
|
|
</ul>
|
|
<br>
|
|
|
|
<a name="FBAHAevents"></a>
|
|
<b>Generierte Events:</b>
|
|
<ul>
|
|
<li>UNDEFINED FBDECT_$ahaName_${NR} FBDECT $id"
|
|
</li>
|
|
</ul>
|
|
|
|
</ul>
|
|
=end html_DE
|
|
|
|
=cut
|